9

所以我注意到这段代码有效:

class Program
{
    public static void Main()
    {
        Int32[ ]numbers = {1,2,3,4,5};

        using (var enumerator = Data().GetEnumerator())
        {

        }
    }

    public static IEnumerable<String> Data()
    {
        yield return "Something";
    }
}

特别是,我对这个using街区很好奇,因为:

Int32[] numbers = { 1, 2, 3, 4, 5, 6 };

using (var enumerator = numbers.GetEnumerator())
{

}

因编译器错误而失败。显然,yield return返回的类是IDisposable常规数组枚举器,而不是。所以现在我很好奇:究竟yield return创造了什么?

4

1 回答 1

11

IEnumerator<T>实现IDisposable,如您在对象浏览器或 MSDN 中所见。

非泛型IEnumerator没有。

Array类实现IEnumerable但不实现IEnumerable<T>。(因为Array不是通用的)
具体的数组类型确实实现了IEnumerable<T>,但它们GetEnumerator()显式地实现了(我不确定为什么)。
因此,GetEnumerator()任何数组类型上的可见都返回IEnumerator.

通用IEnumerable<T>实现返回一个System.SZArrayHelper.SZGenericArrayEnumerator<T>.

此类 (in Array.cs) 的源代码具有以下注释,部分解释了这一点(请记住,对泛型数组的所有支持都可以追溯到IEnumerable<T>不违反规则的时代)

//--------------------------------------------------------------------------------------- 
// ! READ THIS BEFORE YOU WORK ON THIS CLASS. 
//
// The methods on this class must be written VERY carefully to avoid introducing security holes. 
// That's because they are invoked with special "this"! The "this" object
// for all of these methods are not SZArrayHelper objects. Rather, they are of type U[]
// where U[] is castable to T[]. No actual SZArrayHelper object is ever instantiated. Thus, you will
// see a lot of expressions that cast "this" "T[]". 
//
// This class is needed to allow an SZ array of type T[] to expose IList<T>, 
// IList<T.BaseType>, etc., etc. all the way up to IList<Object>. When the following call is 
// made:
// 
//   ((IList<T>) (new U[n])).SomeIListMethod()
//
// the interface stub dispatcher treats this as a special case, loads up SZArrayHelper,
// finds the corresponding generic method (matched simply by method name), instantiates 
// it for type <T> and executes it.
// 
// The "T" will reflect the interface used to invoke the method. The actual runtime "this" will be 
// array that is castable to "T[]" (i.e. for primitivs and valuetypes, it will be exactly
// "T[]" - for orefs, it may be a "U[]" where U derives from T.) 
//---------------------------------------------------------------------------------------
于 2013-03-11T15:05:30.577 回答