查看原文
其他

如何在 C#9 中使用 static 匿名函数

DotNet 2021-09-23

The following article is from 码农读书 Author 码农读书

(给DotNet加星标,提升.Net技能


匿名函数 在 C# 中已经出现很多年了,虽然匿名函数用起来很爽,但代价是不小的,为了避免不必要那些你意想不到的内存分配,这就是为什么 C#9 中引入 static 匿名函数的原因,这篇文章我们将会讨论如何使用 静态匿名函数 以及为什么要用。


匿名方法的代价

匿名方法代价不低,因为它有 委托调用 方面的开销,什么意思呢?如果你的 lambda 里需要捕获封闭方法的局部变量或者参数,那么就会存在两种堆分配,一种是委托上的分配,另一种是闭包上的分配,如果你的 lambda 仅仅捕获一个封闭方法的实例状态,那么仅会有委托分配,如果你的 lambda 什么都不捕获或者仅捕获一个静态状态,那么就没有任何分配。如果你有点懵的话,使用下面的例子加深理解。

int y = 1;  

上面代码的 lambda 中需要获取 y,所以就有了意想不到的堆分配,要想解决,可以将 y 定义为 const 或者 static 来避免这种不必要的分配开销,修改代码如下:

const int y = 1;  
MyMethod(static x => x + y);  
  

为了避免这种不必要和浪费内存的分配,可以在 lambda 上使用 static 关键词 及 变量上标注 const,值得注意的是,static 匿名函数不能访问封闭方法的局部变量和参数和 this 指针,但可以引用它的 静态方法常量

C#9 中使用静态匿名方法

先看一下例子:

    public class Demo  
    {  
        private string formattedText = "{0} It was developed by Microsoft's Anders Hejlsberg in the year 2000.";  
        void DisplayText(Func<string, string> func)  
        {  
            Console.WriteLine(func("C# is a popular programming language."));  
        }  
        public void Display()  
        {  
            DisplayText(text => string.Format(formattedText, text));  
            Console.Read();  
        }  
    }  
    class Program  
    {  
        static void Main(string[] args)  
        {  
            new Demo().Display();  
            Console.Read();  
        }  
    }  

上面的例子中,formattedText变量会被 DisplayText 方法中的 func 所捕获,这也就意味着它会产生你意料之外的内存分配,把程序跑起来之后,会看到如下的输出。

为了避免这种内存分配,你只需要做两件事。

  • formattedText 上标记 const。
  • lambda 上标记 static。

所以修改后的代码如下:

    public class Demo  
    {  
        private const string formattedText = "{0} It was developed by Microsoft's Anders Hejlsberg in the year 2000.";  
        void DisplayText(Func<string, string> func)  
        {  
            Console.WriteLine(func("C# is a popular programming language."));  
        }  
        public void Display()  
        {  
            DisplayText(static text => string.Format(formattedText, text));  
            Console.Read();  
        }  
    }  

现在就没有任何你意料之外的分配了,我想这也是你想要的。现在你可以使用 static + const 组合来提升应用程序性能了,同时也可以有效的阻止在 lambda 中误用封闭方法中的局部变量和参数引发的不必要开销。


- EOF -


推荐阅读  点击标题可跳转
.NET 与树莓派:小风扇模块.NET 中使用 Flurl 高效处理Http请求快速了解C# 8.0 中 "可空引用类型" 语言特性


看完本文有收获?请转发分享给更多人

推荐关注「DotNet」,提升.Net技能 

点赞和在看就是最大的支持❤️

: . Video Mini Program Like ,轻点两下取消赞 Wow ,轻点两下取消在看

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存