天天看點

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

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

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

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

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");      

  使用舉例:

ListCitys = new List()
            {
               "BeiJing",
               "ShangHai",
               "Tianjin",
               "GuangDong"
            };
            var  result1 = Citys.First(c => c.Length > 7);
            Console.WriteLine(result1);
            var result2 = Citys.Where(c => c.Length > 7);
            foreach(var r in result2)
            {
                Console.WriteLine(r);
            }
            Console.ReadKey();      

  

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

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

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

如果我們自己寫,還要寫循環周遊集合,然後判斷字元串長度是否大于7,起碼要寫四五行代碼,而這裡隻要一行就夠了,而且LINQ也要寫很長。

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

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

   public void LambdaFun(string str,Funcfunc)
      {
         Console.WriteLine(func(str));
      }      

  這裡用到了Func

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

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

    總結下紅色那段話的意思:微軟告訴你:我們在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);
        }

        ////// 加法
        ////////////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);
            //
            Listpeople = LoadData();//初始化
            IEnumerableresults = people.Where(delegate(People p) { return p.age > 20; });
        }

        private static ListLoadData()
        {
            Listpeople = new List();   //建立泛型對象  
            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 是參數類型,這是一個泛型類型的委托,用起來很友善的。

 先上例子

 static void Main(string[] args)
        {
            Funcgwl = 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)
        {
            Funcgwl = (p, j) =>
                {
                    if (p + j == 10)
                    {
                        return true;
                    }
                    return false;
                };
            Console.WriteLine(gwl(5,5) + "");   //列印‘True’,z對應參數b,p對應參數a
            Console.ReadKey();
        }      

繼續閱讀