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

  使用舉例:

List<string> Citys = new List<string>()
      {
        "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();

  

這個是大傢熟悉的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表達式的強大瞭吧。

總結下紅色那段話的意思:微軟告訴你:我們在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也就沒什麼瞭!

到此這篇關於C# Lambda表達式及Lambda表達式樹的創建過程的文章就介紹到這瞭,更多相關C# Lambda表達式內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: