(轉(zhuǎn))C# Lambda表達(dá)式詳解,及Lambda表達(dá)式樹的創(chuàng)建

最近由于項(xiàng)目需要,剛剛學(xué)完了Action委托和Func<T>委托,發(fā)現(xiàn)學(xué)完了委托就必須學(xué)習(xí)lambda表達(dá)式,委托和Lambda表達(dá)式聯(lián)合起來,才能充分的體現(xiàn)委托的便利、才能使代碼更加簡(jiǎn)介、優(yōu)雅。

Lambda表達(dá)式

"Lambda表達(dá)式"是一個(gè)匿名函數(shù),是一種高效的類似于函數(shù)式編程的表達(dá)式,Lambda簡(jiǎn)化了開發(fā)中需要編寫的代碼量。它可以包含表達(dá)式和語句,并且可用于創(chuàng)建委托或表達(dá)式目錄樹類型,支持帶有可綁定到委托或表達(dá)式樹的輸入?yún)?shù)的內(nèi)聯(lián)表達(dá)式。所有Lambda表達(dá)式都使用Lambda運(yùn)算符=>,該運(yùn)算符讀作"goes to"。Lambda運(yùn)算符的左邊是輸入?yún)?shù)(如果有),右邊是表達(dá)式或語句塊。Lambda表達(dá)式x => x * x讀作"x goes to x times x"??梢詫⒋吮磉_(dá)式分配給委托類型,如下所示:

delegate int del(int i);
static void Main(string[] args)
{
    del myDelegate = x => x * x;
    int j = myDelegate(5); //j = 25
}

若要?jiǎng)?chuàng)建表達(dá)式目錄樹類型(后面會(huì)詳細(xì)說明):

using System.Linq.Expressions;

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            Expression<del> myET = x => x * x;
        }
    }
}

1、表達(dá)式Lambda

表達(dá)式位于 => 運(yùn)算符右側(cè)的 lambda 表達(dá)式稱為“表達(dá)式 lambda”。 表達(dá)式 lambda 會(huì)返回表達(dá)式的結(jié)果,并采用以下基本形式:

(input parameters) => expression

僅當(dāng) lambda 只有一個(gè)輸入?yún)?shù)時(shí),括號(hào)才是可選的;否則括號(hào)是必需的。 括號(hào)內(nèi)的兩個(gè)或更多輸入?yún)?shù)使用逗號(hào)加以分隔:

(x, y) => x == y

有時(shí),編譯器難以或無法推斷輸入類型。 如果出現(xiàn)這種情況,你可以按以下示例中所示方式顯式指定類型:

(int x, string s) => s.Length > x

使用空括號(hào)指定零個(gè)輸入?yún)?shù):

() => SomeMethod()

2、語句Lambda
當(dāng)lambda表達(dá)式中,有多個(gè)語句時(shí),寫成如下形式:

(input parameters) => {statement;}

例如:

delegate void TestDelegate(string s);
…
TestDelegate myDel = n => { string s = n + " " + "World"; Console.WriteLine(s); };
myDel("Hello");

看到這里,Lambda的基礎(chǔ)知識(shí)就學(xué)完了,下面來講解一下實(shí)際中是如何運(yùn)用的,這里寫了幾個(gè)小例子:

            List<string> Citys= new List<string>()
            {
               "BeiJing",
               "ShangHai",
               "Tianjin",
               "GuangDong"
            };
            var result = Citys.First(c => c.Length > 7);

這個(gè)是大家熟悉的LINQ語句,如果沒學(xué)過沒關(guān)系,這里用的只是很簡(jiǎn)單的幾個(gè)方法,相信大家都能看懂。

首先定義一個(gè)Citys集合,初始化有一些數(shù)據(jù)。然后調(diào)用LINQ的first方法,查詢出來長(zhǎng)度大于7的第一個(gè)結(jié)果,看到了吧,這里用的就是Lambda表達(dá)式,

如果我們自己寫,還要寫循環(huán)遍歷集合,然后判斷字符串長(zhǎng)度是否大于7,起碼要寫四五行代碼,而這里只要一行就夠了,而且LINQ也要寫很長(zhǎng)。

這里用的是最簡(jiǎn)單的Lambda表達(dá)式,(input parameters) => expression的形式。

下面來看一下,如何自己定義和使用Lambda表達(dá)式,首先寫下面一個(gè)函數(shù):

    public void LambdaFun(string str,Func<string,string> func)
      {
         Console.WriteLine(func(str));
      }

這里用到了Func<T>委托,不懂的可以去百度查資料,這個(gè)方法什么都沒有做,只是調(diào)用了委托方法,并將參數(shù)傳遞過去,下面來看一下使用方法:

        LambdaFun("BeiJing 2013", s => 
         {
            if (s.Contains("2013"))
            {
               s = s.Replace("2013", "2014");
            }
            return s;
         });

這里將傳入字符串中的2013替換成為2014,當(dāng)然還可以是將其他字符串替換城任何內(nèi)容,或者是截取,連接等等,完全由我們傳入的Lambda表達(dá)式?jīng)Q定,到了這里感覺到Lambda表達(dá)式的強(qiáng)大了吧。

**lambda表達(dá)式樹動(dòng)態(tài)創(chuàng)建方法 **

        static void Main(string[] args)
        {
            //i*j+w*x
            ParameterExpression a = Expression.Parameter(typeof(int),"i");   //創(chuàng)建一個(gè)表達(dá)式樹中的參數(shù),作為一個(gè)節(jié)點(diǎn),這里是最下層的節(jié)點(diǎn)
            ParameterExpression b = Expression.Parameter(typeof(int),"j");
            BinaryExpression r1 = Expression.Multiply(a,b);    //這里i*j,生成表達(dá)式樹中的一個(gè)節(jié)點(diǎn),比上面節(jié)點(diǎn)高一級(jí)

            ParameterExpression c = Expression.Parameter(typeof(int), "w");
            ParameterExpression d = Expression.Parameter(typeof(int), "x");
            BinaryExpression r2 = Expression.Multiply(c, d);

            BinaryExpression result = Expression.Add(r1,r2);   //運(yùn)算兩個(gè)中級(jí)節(jié)點(diǎn),產(chǎn)生終結(jié)點(diǎn)

            Expression<Func<int, int, int, int, int>> lambda = Expression.Lambda<Func<int, int, int, int, int>>(result,a,b,c,d);

            Console.WriteLine(lambda + "");   //輸出‘(i,j,w,x)=>((i*j)+(w*x))’,z對(duì)應(yīng)參數(shù)b,p對(duì)應(yīng)參數(shù)a

            Func<int, int, int, int, int> f= lambda.Compile();  //將表達(dá)式樹描述的lambda表達(dá)式,編譯為可執(zhí)行代碼,并生成該lambda表達(dá)式的委托;

            Console.WriteLine(f(1, 1, 1, 1) + "");  //輸出結(jié)果2
            Console.ReadKey();
        }

為了便于大家理解,這點(diǎn)代碼構(gòu)成的Lambda表達(dá)式樹如下圖:


image

其實(shí)Lambda表達(dá)式并不難,只有理解了其中的原理,還是很快可以上手的!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容