欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C# 本地函數(shù)與 Lambda 表達(dá)式詳細(xì)介紹

 更新時(shí)間:2021年09月08日 10:49:13   作者:Vladimir Sadov  
這篇文章主要介紹了C 語(yǔ)言本地函數(shù)與 Lambda 表達(dá)式,,由于 C# 長(zhǎng)期以來(lái)一直使用 lambda,因此從差異和相似之處來(lái)看本地函數(shù)確實(shí)是有意義的,感興趣的小伙伴可以參考下面文章內(nèi)容

1、C# 本地函數(shù)與 Lambda 表達(dá)式

C# 局部函數(shù)通常被視為 lambda 表達(dá)式的進(jìn)一步增強(qiáng)。雖然功能是相關(guān)的,但也存在重大差異。

Local Functions嵌套函數(shù)]功能的 C# 實(shí)現(xiàn)。一種語(yǔ)言在支持 lambdas 之后獲得對(duì)嵌套函數(shù)的支持幾個(gè)版本是有點(diǎn)不尋常的。通常情況相反。

Lambda 或一般的一流函數(shù)需要實(shí)現(xiàn)未在堆棧上分配且生命周期與需要它們的功能對(duì)象相關(guān)聯(lián)的局部變量。如果不依賴(lài)?yán)占蛲ㄟ^(guò)捕獲列表等解決方案將變量所有權(quán)的負(fù)擔(dān)減輕給用戶(hù),則幾乎不可能正確有效地實(shí)現(xiàn)它們。對(duì)于某些早期語(yǔ)言來(lái)說(shuō),這是一個(gè)嚴(yán)重的阻塞問(wèn)題。嵌套函數(shù)的簡(jiǎn)單實(shí)現(xiàn)不會(huì)遇到這種復(fù)雜情況,因此一種語(yǔ)言更常見(jiàn)的是僅支持嵌套函數(shù)而不支持 lambda。

無(wú)論如何,由于 C# 長(zhǎng)期以來(lái)一直使用 lambda,因此從差異和相似之處來(lái)看本地函數(shù)確實(shí)是有意義的。

2、Lambda 表達(dá)式

Lambda 表達(dá)式x => x + x是抽象地表示一段代碼以及它如何綁定到其詞法環(huán)境中的參數(shù)和變量的表達(dá)式。作為代碼的抽象表示,lambda 表達(dá)式不能單獨(dú)使用。為了使用由 lambda 表達(dá)式生成的值,需要將其轉(zhuǎn)換為更多內(nèi)容,例如委托或表達(dá)式樹(shù)。

using System;
using System.Linq.Expressions;

class Program
{
    static void Main(string[] args)
    {
        // can't do much with the lambda expression directly
        // (x => x + x).ToString();  // error

        // can assign to a variable of delegate type and invoke
        Func<int, int> f = (x => x + x);
        System.Console.WriteLine(f(21)); // prints "42"

        // can assign to a variable of expression type and introspect
        Expression<Func<int, int>> e = (x => x + x);
        System.Console.WriteLine(e);     // prints "x => (x + x)"
    }
}

有幾點(diǎn)值得注意:

  • lambdas 是產(chǎn)生函數(shù)值的表達(dá)式。
  • lambda 值的生命周期是無(wú)限的——從 lambda 表達(dá)式的執(zhí)行開(kāi)始,只要存在對(duì)該值的任何引用。這意味著 lambda 從封閉方法中使用或“捕獲”的任何局部變量都必須在堆上分配。由于 lambda 值的生命周期不受產(chǎn)生它的堆棧幀的生命周期的限制,因此不能在該堆棧幀上分配變量。
  • lambda 表達(dá)式要求在執(zhí)行 lambda 表達(dá)式時(shí)明確分配主體中使用的所有外部變量。lambda 的第一次和最后一次使用的時(shí)刻很少是確定性的,因此該語(yǔ)言假設(shè) lambda 值可以在創(chuàng)建后立即使用,只要它們是可訪(fǎng)問(wèn)的。因此,一個(gè) lambda 值在創(chuàng)建時(shí)必須是完全可用的,并且它使用的所有外部變量都必須明確分配。
        int x;

        // ERROR: 'x' is not definitely assigned
        Func<int> f = () => x;
  • lambdas 沒(méi)有名字,也不能被象征性地引用。特別是 lambda 表達(dá)式不能遞歸聲明。

注意:可以通過(guò)調(diào)用分配給 lambda 的變量或傳遞給自應(yīng)用其參數(shù)的高階方法來(lái)創(chuàng)建遞歸 lambda,但這不會(huì)表達(dá)真正的自我參照。

3、本地函數(shù)

局部函數(shù)基本上只是在另一個(gè)方法中聲明的方法,作為一種降低方法對(duì)其聲明范圍內(nèi)的可見(jiàn)性的方法。

自然地,局部函數(shù)中的代碼可以訪(fǎng)問(wèn)其包含范圍內(nèi)可訪(fǎng)問(wèn)的所有內(nèi)容——局部變量、封閉方法的參數(shù)、類(lèi)型參數(shù)、局部函數(shù)。一個(gè)值得注意的例外是外部方法標(biāo)簽的可見(jiàn)性。封閉方法的標(biāo)簽在局部函數(shù)中不可見(jiàn)。這只是普通的詞法范圍,它的工作原理與 lambdas 相同。

public class C
{
    object o;

    public void M1(int p)
    {
        int l = 123;

        // lambda has access to o, p, l,
        Action a = ()=> o = (p + l);
    }

    public void M2(int p)
    {
        int l = 123;

        // Local Function has access to o, p, l,
        void a()
        {
          o = (p + l);
        }
    }
}

與 lambda 的明顯區(qū)別在于局部函數(shù)具有名稱(chēng)并且可以在沒(méi)有任何間接方式的情況下使用。局部函數(shù)可以是遞歸的。

static int Fac(int arg)
{
    int FacRecursive(int a)
    {
        return a <= 1 ?
                    1 :
                    a * FacRecursive(a - 1);
    }

    return FacRecursive(arg);
}

與 lambda 表達(dá)式的主要語(yǔ)義區(qū)別在于局部函數(shù)不是表達(dá)式,它們是聲明語(yǔ)句。在代碼執(zhí)行方面,聲明是非常被動(dòng)的實(shí)體。事實(shí)上,聲明并沒(méi)有真正被“執(zhí)行”。與標(biāo)簽等其他聲明類(lèi)似,局部函數(shù)聲明只是將函數(shù)引入包含范圍,而無(wú)需運(yùn)行任何代碼。

更重要的是,無(wú)論是聲明本身還是嵌套函數(shù)的常規(guī)調(diào)用都不會(huì)導(dǎo)致對(duì)環(huán)境的不確定捕獲。在簡(jiǎn)單和常見(jiàn)的情況下,如普通的調(diào)用/返回場(chǎng)景,捕獲的局部變量不需要進(jìn)行堆分配。

例子:

public class C
{    
    public void M()
    {
        int num = 123;

        // has access to num
        void  Nested()
        {
           num++;
        }

        Nested();

        System.Console.WriteLine(num);
    }
}

上面的代碼大致相當(dāng)于(反編譯):

public class C
{
  // A struct to hold "num" variable.
  // We are not storing it on the heap,
  // so it does not need to be a class
  private struct <>c__DisplayClass0_0
  {
      public int num;
  }

  public void M()
  {
      // reserve storage for "num" in a display struct on the _stack_
      C.<>c__DisplayClass0_0 env = default(C.<>c__DisplayClass0_0);

      // num = 123
      env.num = 123;

      // Nested()
      // note - passes env as an extra parameter
      C.<M>g__a0_0(ref env);

      // System.Console.WriteLine(num)
      Console.WriteLine(env.num);
  }

    // implementation of the the "Nested()".
    // note - takes env as an extra parameter
    // env is passed by reference so it's instance is shared
    // with the caller "M()"
    internal static void <M>g__a0_0(ref C.<>c__DisplayClass0_0 env)
    {
        env.num += 1;
    }
}

請(qǐng)注意,上面的代碼直接調(diào)用了“Nested()”的實(shí)現(xiàn)(不是通過(guò)委托間接),并且沒(méi)有在堆上引入顯示存儲(chǔ)的分配(就像 lambda 會(huì)那樣)。局部變量存儲(chǔ)在結(jié)構(gòu)中而不是類(lèi)中。的生命周期num并沒(méi)有因?yàn)樗?中的使用而改變Nested(),所以它仍然可以在棧上分配。M()可以只通過(guò)num引用傳遞,但編譯器使用結(jié)構(gòu)體進(jìn)行打包,因此它可以傳遞所有本地變量,就像num只使用一個(gè) env 參數(shù)一樣。

另一個(gè)有趣的一點(diǎn)是,只要本地函數(shù)在給定范圍內(nèi)可見(jiàn),就可以使用它們。這是一個(gè)重要的事實(shí),使遞歸和相互遞歸的場(chǎng)景成為可能。這也使得本地函數(shù)聲明在源代碼中的確切位置在很大程度上變得不重要。

例如,封閉方法的所有變量必須在調(diào)用讀取它們的本地函數(shù)時(shí)明確分配,而不是在其聲明時(shí)。實(shí)際上,如果調(diào)用可以更早發(fā)生,那么在聲明時(shí)提出該要求將沒(méi)有任何好處。

public void M()
{
    // error here -
    // Use of unassigned local variable 'num'
    Nested();

    int num;

    // whether 'num' is assigned here or not is irrelevant
    void  Nested()
    {
       num++;
    }

    num = 123;

    // no error here - 'num' is assigned
    Nested();

    System.Console.WriteLine(num);
}

此外 - 如果從未使用過(guò)局部函數(shù),它也不會(huì)比一段無(wú)法訪(fǎng)問(wèn)的代碼和任何變量更好,否則它會(huì)使用,不需要分配。

public void M()
{        
    int num;

    // warning - Nested() is never used.
    void  Nested()
    {
       // no errors on unassigned 'num'.
       // this code never runs.
       num++;
    }
}

4、那么,局部函數(shù)的目的是什么?

與 lambdas 相比,局部函數(shù)的主要價(jià)值主張是局部函數(shù)在概念上和運(yùn)行時(shí)開(kāi)銷(xiāo)方面都更簡(jiǎn)單。

Lambda 可以很好地充當(dāng)一類(lèi)函數(shù)的角色,但有時(shí)您只需要一個(gè)簡(jiǎn)單的助手。分配給局部變量的 Lambda 可以完成這項(xiàng)工作,但存在間接開(kāi)銷(xiāo)、委托分配和可能的閉包開(kāi)銷(xiāo)。私有方法也有效,調(diào)用成本更低,但存在封裝問(wèn)題,或缺乏封裝。這樣的助手對(duì)包含類(lèi)型中的每個(gè)人都是可見(jiàn)的。太多這樣的幫手會(huì)導(dǎo)致嚴(yán)重的混亂。

局部函數(shù)非常適合這種情況。調(diào)用本地函數(shù)的開(kāi)銷(xiāo)與調(diào)用私有方法的開(kāi)銷(xiāo)相當(dāng),但使用其他不應(yīng)調(diào)用的方法污染包含類(lèi)型沒(méi)有問(wèn)題。

到此這篇關(guān)于C# 本地函數(shù)與 Lambda 表達(dá)式詳細(xì)介紹的文章就介紹到這了,更多相關(guān)C# 本地函數(shù)與 Lambda 表達(dá)式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 淺談C#下winform和JS的互相調(diào)用和傳參(webbrowser)

    淺談C#下winform和JS的互相調(diào)用和傳參(webbrowser)

    下面小編就為大家?guī)?lái)一篇淺談C#下winform和JS的互相調(diào)用和傳參(webbrowser)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2016-12-12
  • c#分頁(yè)讀取GB文本文件實(shí)例

    c#分頁(yè)讀取GB文本文件實(shí)例

    這篇文章主要介紹了c#分頁(yè)讀取GB文本文件的方法,對(duì)于C#實(shí)現(xiàn)對(duì)大數(shù)據(jù)量的讀取非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2014-11-11
  • C#使用集合實(shí)現(xiàn)二叉查找樹(shù)

    C#使用集合實(shí)現(xiàn)二叉查找樹(shù)

    這篇文章介紹了C#使用集合實(shí)現(xiàn)二叉查找樹(shù)的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-08-08
  • C#中winform中panel重疊無(wú)法顯示問(wèn)題的解決

    C#中winform中panel重疊無(wú)法顯示問(wèn)題的解決

    這篇文章主要介紹了C#中winform中panel重疊無(wú)法顯示問(wèn)題的解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-10-10
  • C#面向?qū)ο缶幊讨虚_(kāi)閉原則的示例詳解

    C#面向?qū)ο缶幊讨虚_(kāi)閉原則的示例詳解

    在面向?qū)ο缶幊讨?,SOLID?是五個(gè)設(shè)計(jì)原則的首字母縮寫(xiě),旨在使軟件設(shè)計(jì)更易于理解、靈活和可維護(hù)。本文將通過(guò)實(shí)例詳細(xì)講講C#面向?qū)ο缶幊讨虚_(kāi)閉原則,需要的可以參考一下
    2022-07-07
  • C#反射應(yīng)用實(shí)例

    C#反射應(yīng)用實(shí)例

    這篇文章主要介紹了C#反射應(yīng)用,實(shí)例分析了通過(guò)反射實(shí)現(xiàn)多系統(tǒng)數(shù)據(jù)庫(kù)的配置方法,是比較實(shí)用的技巧,需要的朋友可以參考下
    2014-12-12
  • C#如何使用DateTime.Now.AddDays方法獲取任一天的信息

    C#如何使用DateTime.Now.AddDays方法獲取任一天的信息

    使用DateTime.Now屬性可以得到當(dāng)前的日期信息,此時(shí)調(diào)用ToString方法,并在該方法中添加指定的格式化字符串,可以按照要求輸出當(dāng)前日期的信息,本文介紹C#使用DateTime.Now.AddDays方法獲取任一天的信息,感興趣的朋友一起看看吧
    2024-01-01
  • WinForm單例窗體用法實(shí)例

    WinForm單例窗體用法實(shí)例

    這篇文章主要介紹了WinForm單例窗體,結(jié)合實(shí)例形式分析了窗體的單例模式定義、實(shí)現(xiàn)與使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2016-07-07
  • C#通過(guò)正則表達(dá)式實(shí)現(xiàn)提取網(wǎng)頁(yè)中的圖片

    C#通過(guò)正則表達(dá)式實(shí)現(xiàn)提取網(wǎng)頁(yè)中的圖片

    本文給大家分享的是使用C#通過(guò)正則表達(dá)式來(lái)實(shí)現(xiàn)提取網(wǎng)頁(yè)中的圖片的代碼,十分的方便,有需要的小伙伴可以參考下。
    2015-12-12
  • C#單線(xiàn)程和多線(xiàn)程端口掃描器詳解

    C#單線(xiàn)程和多線(xiàn)程端口掃描器詳解

    這篇文章主要為大家詳細(xì)介紹了C#單線(xiàn)程和多線(xiàn)程端口掃描器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-07-07

最新評(píng)論