如何使用枚举器模式(Enumerator pattern):
1、 如果你要实现IEnumerable.GetEnumerator,你也要实现一个GetEnumerator方法的非虚方法版本。你的枚举器类的IEnumerable.GetEnumerator方法应该调用这个非虚方法。如下所示应该返回一个内嵌的枚举器结构:
· class MyClass : IEnumerable
· {
· // non-virtual implementation for your custom collection
· public MyEnumerator GetEnumerator() {
· return new MyEnumerator(this); // Return nested public struct
· }
· // IEnumerator implementation
· public IEnumerator.GetEnumerator() {
· return GetEnumerator();//call the non-interface method
· }
· }
如果你的类已经显示地提供了非虚的GetEnumerator方法,那么foreach语句就会调用这个非虚的GetEnumerator方法。否则,它会调用IEnumerable.GetEnumerator方法,如果你的类实现了Ienumerable接口。调用一个非虚方法比通过接口调用一个虚方法要更有效。
2、 在枚举器结构中显示的实现IEnumerator.Current属性。.NET的实现类在实现时会使得该属性返回一个System.Object对象,而不是一个强类型的对象。这会存在类型转换开销。通过在你的Current属性中返回一个强类型对象或者精确的值类型而不是System.Object,可以避免类型转换的开销。因为你已经显示地实现了一个非虚地GetEnumerator方法(不是IEnumerable.GetEnumerator方法),CLR能够直接调用Enumerator.Current属性,而不是调用IEnumerator.Current属性,因此可以直接获取所期待的数据,还能避免类型转换和装箱操作的开销。而减少虚方法调用,可以利用方法的内联了。你的实现应该如下:
// Custom property in your class
//call this property to avoid the boxing or casting overhead
Public MyValueType Current {
MyValueType obj = new MyValueType();
// the obj fields are populated here
return obj;
}
// Explicit member implementation
Object IEnumerator.Current {
get { return Current} // Call the non-interface property to avoid casting
}
注:枚举器模式只有当性能是关键时才适用。
下面的示例代码展示了枚举器模式:
public class ItemTypeCollection: IEnumerable
{
public struct MyEnumerator : IEnumerator
{
public ItemType Current { get {… } }
object IEnumerator.Current { get { return Current; } }
public bool MoveNext() { … }
… }
public MyEnumerator GetEnumerator() { … }
IEnumerator IEnumerable.GetEnumerator() { … }
…}
要利用JIT的内联方式,应该避免在你的集合类中使用虚方法,除非你确实需要扩展。同时,在Current属性的实现中返回当前值来允许使用内联,或者使用一个字段。