为了账号安全,请及时绑定邮箱和手机立即绑定

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

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

C#
炎炎设计 2019-09-03 17:11:26
我一直在学习C#,我正在努力理解lambdas。在下面的示例中,它打印出10次。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语法[&](){ ... } // Capture by reference与[=](){ ... } // Capture copies
查看完整描述

3 回答

?
幕布斯7119047

TA贡献1794条经验 获得超8个赞

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


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


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循环中创建副本,编译器在每次迭代中生成新对象,如下所示:


for (int i = 0; i < 10; ++i)

{

    DisplayClass1 displayClass1 = new DisplayClass1();

    displayClass1.i = i;

    actions.Add(new Action(displayClass1.Lambda));

}


查看完整回答
反对 回复 2019-09-03
?
呼如林

TA贡献1798条经验 获得超3个赞

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


for (int i = 0; i < 10; ++i)

{

    int copy = i;

    actions.Add(() => Console.WriteLine(copy));

}

但是我无法理解为什么在for-loop中放置副本与使用lambda捕获有什么不同i。


查看完整回答
反对 回复 2019-09-03
  • 3 回答
  • 0 关注
  • 480 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信