委託、Lambda表示式、事件系列03,從委託到Lamda表示式

Darren Ji發表於2014-10-01

在"委託、Lambda表示式、事件系列02,什麼時候該用委託"一文中,使用委託讓程式碼簡潔了不少。

namespace ConsoleApplication2
{
    internal delegate bool MyCalculateDelegate(int val);
    class Program
    {
        static void Main(string[] args)
        {
            IEnumerable<int> source = new List<int>(){2, 3, 4, 5, 6, 7, 8, 9,10, 11};
            MyCalculateDelegate del = LessThanFive;
            var result = GetNumbers(source, del);
            foreach (int n in result)
            {
                Console.WriteLine(n); 
            }
        }
        static IEnumerable<int> GetNumbers(IEnumerable<int> numbers, MyCalculateDelegate del)
        {
            foreach (int number in numbers)
            {
                if (del(number)) yield return number;
            }
        }
        static bool LessThanFive(int val)
        {
            return val < 5;
        }
        static bool LessThanTen(int val)
        {
            return val < 10;
        }
    }
}

 

可是,以上LessThanFive方法和LessThanTen方法的輸入引數、輸出型別、以及實現邏輯都是一致的,有沒有一種形式可以替代它們?這正是Lambda表示式登場的時候!Lambda表示式是匿名委託,以=>分隔,左邊是輸入引數,右邊是實現過程。

 

namespace ConsoleApplication2
{
    internal delegate bool MyCalculateDelegate(int val);
    class Program
    {
        static void Main(string[] args)
        {
            IEnumerable<int> source = new List<int>(){2, 3, 4, 5, 6, 7, 8, 9,10, 11};
            var result = GetNumbers(source, n => n < 5);
            foreach (int n in result)
            {
                Console.WriteLine(n); 
            }
        }
        static IEnumerable<int> GetNumbers(IEnumerable<int> numbers, MyCalculateDelegate del)
        {
            foreach (int number in numbers)
            {
                if (del(number)) yield return number;
            }
        }
    }
}

以上,使用Lambda表示式又讓程式碼簡潔了不少!GetNumbers的實參n => n < 5,就是一個Lambda表示式,它符合委託MyCalculateDelegate的定義,輸入引數是整型,輸出是bool型別。

 

其實,lambda表示式n => n < 5是一種"語法糖",內部還是執行了如下程式碼:

        ......
        MyCalculateDelegate del = LessThanFive;
        var result = GetNumbers(source, del);
        ......
        static bool LessThanFive(int val)
        {
            return val < 5;
        }


這一點可以從IL層面看出。使用Reflector進行反編譯:

private static void Main(string[] args)
{
    List<int> <>g__initLocal0 = new List<int>();
    <>g__initLocal0.Add(2);
    <>g__initLocal0.Add(3);
    <>g__initLocal0.Add(4);
    <>g__initLocal0.Add(5);
    <>g__initLocal0.Add(6);
    <>g__initLocal0.Add(7);
    <>g__initLocal0.Add(8);
    <>g__initLocal0.Add(9);
    <>g__initLocal0.Add(10);
    <>g__initLocal0.Add(11);
    IEnumerable<int> source = <>g__initLocal0;
    IEnumerable<int> result = GetNumbers(source, (CS$<>9__CachedAnonymousMethodDelegate2 != null) ? CS$<>9__CachedAnonymousMethodDelegate2 : (CS$<>9__CachedAnonymousMethodDelegate2 = new MyCalculateDelegate(Program.<Main>b__1)));
    foreach (int n in result)
    {
        Console.WriteLine(n);
    }
}

 

以上,由於n => n > 5符合委託MyCalculateDelegate的定義,在內部建立了一個MyCalculateDelegate型別的委託,然後把一個編譯器生成的方法<Main>b__1賦給了委託變數。

 

[CompilerGenerated]
private static bool <Main>b__1(int n)
{
    return (n < 5);
}


<Main>b__1方法是由編譯器幫我們自動生成的。

 

總結:lambda表示式其實是"語法糖",原始碼被編譯的時候,編譯器會自動幫我們生成一個符合委託定義的方法,然後把該方法賦給委託。

 

 

“委託、Lambda表示式、事件系列”包括:

委託、Lambda表示式、事件系列01,委託是什麼,委託的基本用法,委託的Method和Target屬性

委託、Lambda表示式、事件系列02,什麼時候該用委託

委託、Lambda表示式、事件系列03,從委託到Lamda表示式

委託、Lambda表示式、事件系列04,委託鏈是怎樣形成的, 多播委託, 呼叫委託鏈方法,委託鏈異常處理

委託、Lambda表示式、事件系列05,Action委託與閉包

委託、Lambda表示式、事件系列06,使用Action實現觀察者模式,體驗委託和事件的區別

委託、Lambda表示式、事件系列07,使用EventHandler委託

相關文章