c#中的擴展方法學(xué)習(xí)筆記
前言
最近在看王清培前輩的.NET框架設(shè)計時,當(dāng)中有提到擴展方法 .
開頭的一句話是:擴展方法是讓我們在不改變類原有代碼的情況下動態(tài)地添加方法的方式,這給面向?qū)ο笤O(shè)計 模塊設(shè)計帶來了質(zhì)的提升
很明顯,擴展方法在框架設(shè)計或者平時碼代碼中,是能夠提升我們整個架構(gòu)的靈活性的
簡介
擴展方法被定義為靜態(tài)方法,但它們是通過實例方法語法進(jìn)行調(diào)用的。 它們的第一個參數(shù)指定該方法作用于哪個類型,并且該參數(shù)以 this 修飾符為前綴。 擴展方法當(dāng)然不能破壞面向?qū)ο蠓庋b的概念,所以只能是訪問所擴展類的public成員。
擴展方法使您能夠向現(xiàn)有類型“添加”方法,而無需創(chuàng)建新的派生類型、重新編譯或以其他方式修改原始類型。擴展方法是一種特殊的靜態(tài)方法,但可以像擴展類型上的實例方法一樣進(jìn)行調(diào)用。
C#擴展方法第一個參數(shù)指定該方法作用于哪個類型,并且該參數(shù)以 this 修飾符為前綴。
擴展方法的目的就是為一個現(xiàn)有類型添加一個方法,現(xiàn)有類型既可以是int,string等數(shù)據(jù)類型,也可以是自定義的數(shù)據(jù)類型。
一..net自帶擴展方法和自定義擴展方法
在使用linq時就能夠使用到很多.net自帶的擴展方法,比如where select等等
where的擴展方法定義
public static IEnumerable<TSource> Where<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate);
Select的擴展方法定義
public static IEnumerable<TResult> Select<TSource, TResult>(this IEnumerable<TSource> source, Func<TSource, TResult> selector);
(1)自己實現(xiàn)where和select的擴展方法
// where自實現(xiàn)
public static IEnumerable<TSource> ExtenSionWhere<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate)
{
if (source == null)
{
throw new Exception(nameof(source));
}
if (predicate == null)
{
throw new Exception(nameof(predicate));
}
List<TSource> satisfySource = new List<TSource>();
foreach (var sou in source)
{
if (predicate(sou))
{
satisfySource.Add(sou);
}
}
return satisfySource;
}
// select 自實現(xiàn)
public static IEnumerable<TResult> ExtenSionSelect<TSource, TResult>(this IEnumerable<TSource> source, Func<TSource, TResult> selector)
{
if(source==null)
{
throw new Exception(nameof(source));
}
if(selector==null)
{
throw new Exception(nameof(source));
}
List<TResult> resultList = new List<TResult>();
foreach(var sou in source)
{
resultList.Add(selector(sou));
}
return resultList;
}
(2)自實現(xiàn)where和select調(diào)用
static void Main(string[] args)
{
List<int> list = new List<int>() { 1, 2, 3, 4, 5, 6 };
//常規(guī)寫法
var selectList = list.ExtenSionWhere(p => p > 3).ExtenSionSelect(p => p.ToString()).ToList();
//自定義泛型委托寫法
Func<int, bool> whereFunc = (num) => num > 3;
Func<int, string> selectFunc = (num) => num.ToString();
var selectList1 = list.ExtenSionWhere(p => whereFunc(p)).ExtenSionSelect(p => selectFunc(p)).ToList();
}
二.使用擴展方法實現(xiàn)鏈?zhǔn)骄幊?/strong>
我在項目中經(jīng)常使用開源的Flurl進(jìn)行http請求,在進(jìn)行拼裝請求報文時,就會使用到鏈?zhǔn)骄幊?/p>
如下代碼所示
以上代碼就是使用了擴展方法進(jìn)行鏈?zhǔn)骄幊?從而使得整個請求信息可以在一句代碼中體現(xiàn)出來
接下來,我們自己實現(xiàn)鏈?zhǔn)酱a
public static class ContextExtension
{
public static RectangleContext SetLength(this RectangleContext context,int num)
{
RectangleContext.Config.Length = num;
return context;
}
public static RectangleContext SetWideth(this RectangleContext context, int num)
{
RectangleContext.Config.Wideth = num;
return context;
}
public static RectangleContext SetHeight(this RectangleContext context, int num)
{
RectangleContext.Config.Height = num;
return context;
}
}
public class RectangleContext
{
public static RectangleContext Config=new RectangleContext();
public int Length { get; set; }
public int Wideth { get; set; }
public int Height { get; set; }
}
調(diào)用和執(zhí)行結(jié)果

總結(jié)
1.使用擴展方法能在不修改原有類型的基礎(chǔ)下,動態(tài)添加方法,這使得整個框架更具有靈活性
2.在使用上下文信息的時候,可以使用鏈?zhǔn)骄幊?使得調(diào)用時能夠在一句代碼中完成所有屬性設(shè)置
3.擴展方法不能濫用.添加擴展方法應(yīng)當(dāng)使用最小影響原則,即盡量不要在父類使用擴展方法,比如object,這將影響性能
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
Unity3D網(wǎng)格功能生成球體網(wǎng)格模型
這篇文章主要為大家詳細(xì)介紹了Unity3D網(wǎng)格功能生成球體網(wǎng)格模型,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-02-02
ASP.NET MVC 5使用X.PagedList.Mvc進(jìn)行分頁教程(PagedList.Mvc)
這篇文章主要介紹了ASP.NET MVC 5使用X.PagedList.Mvc進(jìn)行分頁教程(原名為PagedList.Mvc),需要的朋友可以參考下2014-10-10
C#中使用Interlocked進(jìn)行原子操作的技巧
使用.NET提供的Interlocked類可以對一些數(shù)據(jù)進(jìn)行原子操作,看起來似乎跟lock鎖一樣,但它并不是lock鎖,它的原子操作是基于CPU本身的,非阻塞的,所以要比lock的效率高2016-12-12

