首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >如何告诉lambda函数捕获一个副本,而不是C#中的引用?

如何告诉lambda函数捕获一个副本,而不是C#中的引用?
EN

Stack Overflow用户
提问于 2009-01-16 20:08:10
回答 4查看 27.2K关注 0票数 40

我一直在学习C#,并试图理解lambdas。在下面的示例中,它打印了10次。

代码语言:javascript
复制
class Program
{
    delegate void Action();
    static void Main(string[] args)
    {
        List<Action> actions = new List<Action>();

        for (int i = 0; i < 10; ++i )
            actions.Add(()=>Console.WriteLine(i));

        foreach (Action a in actions)
            a();
    }
}

显然,lambda后面生成的类存储了一个指向int i变量的引用或指针,并在每次循环迭代时为同一引用分配一个新值。有没有一种方法可以强制lamda获取一个副本,比如C++0x语法

代码语言:javascript
复制
[&](){ ... } // Capture by reference

代码语言:javascript
复制
[=](){ ... } // Capture copies
EN

回答 4

Stack Overflow用户

回答已采纳

发布于 2009-01-16 20:24:23

编译器所做的是将您的lambda和lambda捕获的任何变量拉入编译器生成的嵌套类中。

编译后,您的示例看起来很像这样:

代码语言:javascript
复制
class Program
{
        delegate void Action();
        static void Main(string[] args)
        {
                List<Action> actions = new List<Action>();

                DisplayClass1 displayClass1 = new DisplayClass1();
                for (displayClass1.i = 0; displayClass1.i < 10; ++displayClass1.i )
                        actions.Add(new Action(displayClass1.Lambda));

                foreach (Action a in actions)
                        a();
        }

        class DisplayClass1
        {
                int i;
                void Lambda()
                {
                        Console.WriteLine(i);
                }
        }
}

通过在for循环中创建一个副本,编译器在每次迭代中生成新对象,如下所示:

代码语言:javascript
复制
for (int i = 0; i < 10; ++i)
{
    DisplayClass1 displayClass1 = new DisplayClass1();
    displayClass1.i = i;
    actions.Add(new Action(displayClass1.Lambda));
}
票数 32
EN

Stack Overflow用户

发布于 2009-01-16 20:08:57

我能找到的唯一解决方案是先制作一个本地副本:

代码语言:javascript
复制
for (int i = 0; i < 10; ++i)
{
    int copy = i;
    actions.Add(() => Console.WriteLine(copy));
}

但是我很难理解为什么在for循环中放置一个副本与使用lambda捕获i有什么不同。

票数 29
EN

Stack Overflow用户

发布于 2009-01-16 20:13:06

唯一的解决方案是创建一个本地副本,并在lambda中引用它。当在闭包中访问C# (和VB.Net)中的所有变量时,它们将具有引用语义与复制/值语义。在这两种语言中都无法更改此行为。

注意:它实际上不会作为引用进行编译。编译器将变量提升到闭包类中,并将对"i“的访问重定向到给定闭包类中的字段"i”。不过,通常更容易将其视为引用语义。

票数 11
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/451779

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档