9

如何yield实现的模式lazy loading

4

3 回答 3

13

yield 实现在需要之前不会到达代码。

例如,这段代码:

public IEnumerable<int> GetInts()
{
    yield return 1;
    yield return 2;
    yield return 3;
}

实际上会编译成一个实现的嵌套类,IEnumerable<int>并且它的主体GetInts()将返回该类的一个实例。

使用反射器可以看到:

public IEnumerable<int> GetInts()
{
    <GetInts>d__6d d__d = new <GetInts>d__6d(-2);
    d__d.<>4__this = this;
    return d__d;
}

编辑GetInts- 添加有关实现的更多信息:
此实现使其变得懒惰的方式基于该Enumerator MoveNext()方法。当生成可枚举的嵌套类时(<GetInts>d__6d在示例中),它有一个状态,并且每个状态都有一个值连接(这是一个简单的情况,在更高级的情况下,将在代码达到状态时评估该值)。如果我们看一下MoveNext()代码,<GetInts>d__6d我们会看到状态:

private bool MoveNext()
{
    switch (this.<>1__state)
    {
        case 0:
            this.<>1__state = -1;
            this.<>2__current = 1;
            this.<>1__state = 1;
            return true;

        case 1:
            this.<>1__state = -1;
            this.<>2__current = 2;
            this.<>1__state = 2;
            return true;

        case 2:
            this.<>1__state = -1;
            this.<>2__current = 3;
            this.<>1__state = 3;
            return true;

        case 3:
            this.<>1__state = -1;
            break;
    }
    return false;
}

当枚举器被询问当前对象时,它返回连接到当前状态的对象。

为了表明仅在需要时才评估代码,您可以查看以下示例:

[TestFixture]
public class YieldExample
{
    private int flag = 0;
    public IEnumerable<int> GetInts()
    {
        yield return 1;
        flag = 1;
        yield return 2;
        flag = 2;
        yield return 3;
        flag = 3;
    }

    [Test]
    public void Test()
    {
        int expectedFlag = 0;
        foreach (var i in GetInts())
        {
            Assert.That(flag, Is.EqualTo(expectedFlag));
            expectedFlag++;
        }

        Assert.That(flag, Is.EqualTo(expectedFlag));
    }
}

我希望它更清楚一点。我建议使用 Reflector 查看代码,并在更改“yield”代码时观察编译后的代码。

于 2010-05-03T08:50:44.613 回答
7

如果您想了解更多关于编译器在使用时所做的事情yield return,请查看 Jon Skeet 的这篇文章:Iterator block implementation details

于 2010-05-03T10:15:07.910 回答
4

基本上通过 usingyield语句实现的迭代器被编译成一个实现状态机的类。

如果您从不foreach(=迭代并实际使用)返回IEnumerable<T>的 ,则代码永远不会实际执行。如果这样做,则只执行确定要返回的下一个值所需的最少代码,仅在请求下一个值时恢复执行。

当您在调试器中单步执行此类代码时,您实际上可以看到这种行为发生。至少尝试一次:我认为看到这一步一步发生是很有启发性的。

于 2010-05-03T10:16:16.643 回答