21

场景:循环中的异步任务执行一个方法,该方法包含随着程序继续而变化的参数:

while(this._variable < 100)
{
    this._variable++; 
    var aTask = Task.Factory.StartNew(() =>
    {
        aList.add(this._variable);
        update(this._savePoint);
    });
}

如果循环运行的速度比任务完成的速度快,列表会添加变量的当前值还是变量保存在本地并添加原始值?

4

1 回答 1

20

闭包关闭变量,而不是值。因此,递增_variable 可以改变引用它的任务的行为。

您可以通过制作本地副本来防止这种情况:

while (this._variable < 100)
{
    this._variable++;
    int local = _variable;
    var aTask = Task.Factory.StartNew(() =>
    {
        aList.add(local);
        update(this._savePoint);
    });
} 

或者您可以将值作为状态传递给任务:

while (this._variable < 100)
{
    this._variable++;
    var aTask = Task.Factory.StartNew(object state =>
    {
        aList.add((int)state);
        update(this._savePoint);
    }, this._variable);
} 

这些都通过将值复制_variable到新的临时变量来工作。在第一种情况下,local变量是在循环范围内定义的,因此每次迭代都会得到一个新变量。在第二种情况下,当您将其作为参数_variable传递给 Task 时,您会复制该值。state如果_variable是引用类型,这些解决方案将不起作用;您必须执行克隆。

于 2012-04-16T18:49:52.317 回答