C# Lambda表達式詳解,及Lambda表達式樹的創建


C# Lambda表達式詳解,及Lambda表達式樹的創建

   每次寫博客,第一句話都是這樣的:程序員很苦逼,除了會寫程序,還得會寫博客!當然,希望將來的一天,某位老板看到此博客,給你的程序員職工加點薪資吧!因為程序員的世界除了苦逼就是沉默。我眼中的程序員大多都不愛說話,默默承受着編程的巨大壓力,除了技術上的交流外,他們不願意也不擅長和別人交流,更不樂意任何人走進他們的內心!

   題外話說多了,咱進入正題:

   上一節中,我們講到:在 2.0 之前的 C# 版本中,聲明委托的唯一方法是使用命名方法。  C# 2.0 引入了匿名方法,而在 C# 3.0 及更高版本中,Lambda 表達式取代了匿名方法,作為編寫內聯代碼的首選方式  有一種情況下,匿名方法提供了 Lambda 表達式中所沒有的功能。  您可使用匿名方法來忽略參數列表。  這意味着匿名方法可轉換為具有各種簽名的委托。  這對於 Lambda 表達式來說是不可能的。  有關 lambda 表達式的更多特定信息,請參見 Lambda 表達式(C# 編程指南)。 

    總結下紅色那段話的意思:微軟告訴你:我們在C#2.0之前就有委托了,在2.0之后又引入了匿名方法,C#3.0之后,又引入了Lambda表達式,他們三者之間的順序是:委托->匿名變量->Lambda表達式,微軟的一步步升級,帶給我們編程上的優美,簡潔,可讀性強.....在此,不多誇微軟,怕他們看到這篇博客后驕傲,怕他們尾巴能翹到天上,不知天高地厚。嘿嘿,說多了!

    溫故而知新,可以做老師,咱們來溫故下委托和匿名表達式。

委托如下:

復制代碼
 delegate int calculator(int x, int y); //委托類型
        static void Main()
        {
            calculator cal = new calculator(Adding);
            int He = cal(1, 1);
            Console.Write(He);
        }

        /// <summary>
        /// 加法
        /// </summary>
        /// <param name="x"></param>
        /// <param name="y"></param>
        /// <returns></returns>
        public static int Adding(int x, int y)
        {
            return x + y;
        }
復制代碼

匿名方法如下:

復制代碼
 delegate int calculator(int x, int y); //委托
        static void Main()
        {
            calculator cal = delegate(int num1,int num2)
            {
                return num1 + num2;
            };
            int he = cal(1, 1);
            Console.Write(he);
        }
復制代碼

下面我們來講解Lambda表達式:

 按照上邊的加法,我們用Lambda表達式來實現,代碼如下:

復制代碼
delegate int calculator(int x, int y); //委托類型
        static void Main()
        {
            calculator cal = (x, y) => x + y;//Lambda表達式,大家發現沒有,代碼一個比一個簡潔
            int he = cal(1, 1);
            Console.Write(he);
        }
復制代碼

那么我們詳細講講Lambda表達式:

若要創建 Lambda 表達式,需要在 Lambda 運算符 => 左側指定輸入參數(如果有),然后在另一側輸入表達式或語句塊。 例如,lambda 表達式 x => x * x 指定名為 x 的參數並返回 x 的平方值。 如上面的示例所示,你可以將此表達式分配給委托類型

"Lambda表達式"是一個特殊的匿名函數,是一種高效的類似於函數式編程的表達式,Lambda簡化了開發中需要編寫的代碼量。它可以包含表達式和語句,並且可用於創建委托或表達式目錄樹類型,支持帶有可綁定到委托或表達式樹的輸入參數的內聯表達式。所有Lambda表達式都使用Lambda運算符=>,該運算符讀作"goes to"。Lambda運算符的左邊是輸入參數(如果有),右邊是表達式或語句塊。Lambda表達式x => x * x讀作"x goes to x times x"。舉幾個簡單的Lambda表達式,如下:

復制代碼
delegate bool MyBol(int x, int y);
        delegate bool MyBol_2(int x, string y);
        delegate int calculator(int x, int y); //委托類型
        delegate void VS();
        static void Main()
        {
            MyBol Bol = (x, y) => x == y;
            MyBol_2 Bol_2 = (x, s) => s.Length > x;
            calculator C = (X, Y) => X * Y;
            VS S = () => Console.Write("我是無參數Labada表達式");
            //
            int[] numbers = { 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };
            int oddNumbers = numbers.Count(n => n % 2 == 1);
            //
            List<People> people = LoadData();//初始化
            IEnumerable<People> results = people.Where(delegate(People p) { return p.age > 20; });
        }

        private static List<People> LoadData()
        {
            List<People> people = new List<People>();   //創建泛型對象  
            People p1 = new People(21, "guojing");       //創建一個對象  
            People p2 = new People(21, "wujunmin");     //創建一個對象  
            People p3 = new People(20, "muqing");       //創建一個對象  
            People p4 = new People(23, "lupan");        //創建一個對象  
            people.Add(p1);                     //添加一個對象  
            people.Add(p2);                     //添加一個對象  
            people.Add(p3);                     //添加一個對象  
            people.Add(p4);
            return people;
        }

    }

    public class People
    {
        public int age { get; set; }                //設置屬性  
        public string name { get; set; }            //設置屬性  
        public People(int age, string name)      //設置屬性(構造函數構造)  
        {
            this.age = age;                 //初始化屬性值age  
            this.name = name;               //初始化屬性值name  
        }
    } 
復制代碼

Func<T>委托

 T 是參數類型,這是一個泛型類型的委托,用起來很方便的。

 先上例子

復制代碼
  static void Main(string[] args)
        {
            Func<int, string> gwl = p => p + 10 + "--返回類型為string";            
            Console.WriteLine(gwl(10) + "");   //打印‘20--返回類型為string’,z對應參數b,p對應參數a
            Console.ReadKey();
        }
復制代碼

說明:我們可以看到,這里的p為int 類型參數, 然而lambda主體返回的是string類型的。

再上一個例子

復制代碼
復制代碼
        static void Main(string[] args)
        {
            Func<int, int, bool> gwl = (p, j) =>
                {
                    if (p + j == 10)
                    {
                        return true;
                    }
                    return false;
                };
            Console.WriteLine(gwl(5,5) + "");   //打印‘True’,z對應參數b,p對應參數a
            Console.ReadKey();
        }
復制代碼
復制代碼

說明:從這個例子,我們能看到,p為int類型,j為int類型,返回值為bool類型。

至此,如果上邊的內容都能看懂,那么Lambda也就沒什么了!

@陳卧龍的博客

 
        

出處:

http://www.cnblogs.com/chenwolong/p/5669440.html

Lambda 表達式(C# 編程指南)

==============================================================================================

  最近由於項目需要,剛剛學完了Action委托和Func<T>委托,發現學完了委托就必須學習lambda表達式,委托和Lambda表達式聯合起來,才能充分的體現委托的便利、才能使代碼更加簡介、優雅。

 

Lambda表達式

"Lambda表達式"是一個匿名函數,是一種高效的類似於函數式編程的表達式,Lambda簡化了開發中需要編寫的代碼量。它可以包含表達式和語句,並且可用於創建委托或表達式目錄樹類型,支持帶有可綁定到委托或表達式樹的輸入參數的內聯表達式。所有Lambda表達式都使用Lambda運算符=>,該運算符讀作"goes to"。Lambda運算符的左邊是輸入參數(如果有),右邊是表達式或語句塊。Lambda表達式x => x * x讀作"x goes to x times x"。可以將此表達式分配給委托類型,如下所示:

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

若要創建表達式目錄樹類型(后面會詳細說明):

 
復制代碼
using System.Linq.Expressions;

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

1、表達式Lambda

  表達式位於 => 運算符右側的 lambda 表達式稱為“表達式 lambda”。 表達式 lambda 會返回表達式的結果,並采用以下基本形式:

       (input parameters) => expression

僅當 lambda 只有一個輸入參數時,括號才是可選的;否則括號是必需的。 括號內的兩個或更多輸入參數使用逗號加以分隔:

(x, y) => x == y

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

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

使用空括號指定零個輸入參數:

() => SomeMethod()

2、語句Lambda

當lambda表達式中,有多個語句時,寫成如下形式:

(input parameters) => {statement;}
例如:

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

 

看到這里,Lambda的基礎知識就學完了,下面來講解一下實際中是如何運用的,這里寫了幾個小例子:

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

這個是大家熟悉的LINQ語句,如果沒學過沒關系,這里用的只是很簡單的幾個方法,相信大家都能看懂。

首先定義一個Citys集合,初始化有一些數據。然后調用LINQ的first方法,查詢出來長度大於7的第一個結果,看到了吧,這里用的就是Lambda表達式,

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

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

 

下面來看一下,如何自己定義和使用Lambda表達式,首先寫下面一個函數:

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

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

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

這里將傳入字符串中的2013替換成為2014,當然還可以是將其他字符串替換城任何內容,或者是截取,連接等等,完全由我們傳入的Lambda表達式決定,到了這里感覺到Lambda表達式的強大了吧。

lambda表達式樹動態創建方法 

復制代碼
     static void Main(string[] args)
        {
            //i*j+w*x
            ParameterExpression a = Expression.Parameter(typeof(int),"i");   //創建一個表達式樹中的參數,作為一個節點,這里是最下層的節點
            ParameterExpression b = Expression.Parameter(typeof(int),"j");
            BinaryExpression r1 = Expression.Multiply(a,b);    //這里i*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);   //運算兩個中級節點,產生終結點

            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對應參數b,p對應參數a

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

            Console.WriteLine(f(1, 1, 1, 1) + "");  //輸出結果2
            Console.ReadKey();
        }
復制代碼

為了便於大家理解,這點代碼構成的Lambda表達式樹如下圖:

其實Lambda表達式並不難,只有理解了其中的原理,還是很快可以上手的!

出處:http://www.cnblogs.com/yunfeifei/p/3844814.html

==============================================================================================


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM