Answer:
 

I understand the IEnumerator/IEnumerable methods and properties and also how they are interrelated. But I fail to understand from the foreach loop perspective. Say for example, if I say the following line of code, how does the compiler look at foreach loop and break it down to IEnumerable/IEnumerator for every occurance of foreach loop in your code. In other words, what happens under the hood? I tried to go through couple of articles on the web. But they were NOT looked from the foreach loop perspective or I might have failed to understand. Could somebody explain with the example or direct me to any articles that explains clearly. All help is greatly appreciated.

Answer: To keep it short the compiler doesn't do much. It just translates the foreach code into a while under the hood. Here's an example:

List<int> list = new List<int>;

........

foreach(int item in list)

{

//Do stuff

}

Becomes:

Enumerator<int> enum = ((IEnumerable<int>)list).GetEnumerator();

enum.Reset();

while(enum.MoveNext())

{

int item = enum.Current;

//Do stuff

}

or something very close to that. The acual code compiled most likely has a try / finally block around the while to dispose the enumerator

IEnumerable is the base interface for all non-generic collections that can be enumerated. For the generic version of this interface see System.Collections.Generic.IEnumerable<T>. IEnumerable contains a single method, GetEnumerator, which returns an IEnumerator. IEnumerator provides the ability to iterate through the collection by exposing a Current property and MoveNext and Reset methods.

It is a best practice to implement IEnumerable and IEnumerator on your collection classes to enable the foreach (For Each in Visual Basic) syntax, however implementing IEnumerable is not required. If your collection does not implement IEnumerable, you must still follow the iterator pattern to support this syntax by providing a GetEnumerator method that returns an interface, class or struct. When using Visual Basic, you must provide an IEnumerator implementation, which is returned by GetEnumerator. When developing with C# you must provide a class that contains a Current property, and MoveNext and Reset methods as described by IEnumerator, but the class does not have to implement IEnumerator.

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Collections;
  6.  
  7. namespace CSharpExam
  8. {
  9. public class TestInumerable
  10. {
  11. public static void Run()
  12. {
  13. Person[] peopleArray = new Person[]
  14. {
  15. new Person("Steven", "Xia"),
  16. new Person("Jim", "Johnson"),
  17. new Person("Sue", "Rabon"),
  18. };
  19.  
  20. People peopleList = new People(peopleArray);
  21. foreach (Person p in peopleList)
  22. Console.WriteLine(p.firstName + " " + p.lastName);
  23.  
  24. }
  25. }
  26.  
  27. public class Person
  28. {
  29. public Person(string fName, string lName)
  30. {
  31. this.firstName = fName;
  32. this.lastName = lName;
  33. }
  34.  
  35. public string firstName;
  36. public string lastName;
  37. }
  38.  
  39. public class People //: IEnumerable
  40. {
  41. private Person[] _people;
  42. public People(Person[] pArray)
  43. {
  44. _people = new Person[pArray.Length];
  45.  
  46. for (int i = ; i < pArray.Length; i++)
  47. {
  48. _people[i] = pArray[i];
  49. }
  50. }
  51.  
  52. #region IEnumerable 成员
  53.  
  54. public PeopleEnum GetEnumerator()
  55. {
  56. return new PeopleEnum(_people);
  57. }
  58.  
  59. #endregion
  60. }
  61.  
  62. public class PeopleEnum //: IEnumerator
  63. {
  64. public Person[] _people;
  65.  
  66. // Enumerators are positioned before the first element
  67. // until the first MoveNext() call.
  68. int position = -;
  69.  
  70. public PeopleEnum(Person[] list)
  71. {
  72. _people = list;
  73. }
  74.  
  75. public bool MoveNext()
  76. {
  77. position++;
  78. return (position < _people.Length);
  79. }
  80.  
  81. public void Reset()
  82. {
  83. position = -;
  84. }
  85.  
  86. public object Current
  87. {
  88. get
  89. {
  90. try
  91. {
  92. return _people[position];
  93. }
  94. catch (IndexOutOfRangeException)
  95. {
  96. throw new InvalidOperationException();
  97. }
  98. }
  99. }
  100. }
  101.  
  102. }

随机推荐

  1. Object和Function谁先被创建

    http://bbs.csdn.net/topics/390772104#post-397284029

  2. C#------SortedLIst键值对的使用方法

    方法: SortedList sf = new SortedList(); sf.Add(, "广州"); sf.Add(, "江门"); sf.Add(, & ...

  3. SmartGit Mac、Liunx、Windows过期后破解方法

    根据自己的操作系统,进入相应的文件夹 ,可能还有一个版本号的文件夹,再进入 Windows: %APPDATA%\syntevo\SmartGit\ OS X: ~/Library/Preferenc ...

  4. 在Hyper-V Linux VM如何选择LIS Linux集成服务

    导读 很多工程师都知道,如果你选择在 Hyper-V 中运行 Linux guest VM,要获得最好的使用体验,必需针对你所使用的 Linux 发行版和使用场景选择 Linux Integratio ...

  5. Android StaggeredGrid 加下拉刷新功能 PullToRefresh

    https://github.com/etsy/AndroidStaggeredGrid  用的github上面提供瀑布流,继承于abslistview,回收机制不错,并且提供了OnScrollLis ...

  6. python 学习源

    入门 w3cschool https://www.w3cschool.cn/python/ 菜鸟教程(支持在线编程) http://www.runoob.com/python/python-tutor ...

  7. Spring Security OAuth2 源码分析

    Spring Security OAuth2 主要两部分功能:1.生成token,2.验证token,最大概的流程进行了一次梳理 1.Server端生成token (post /oauth/token ...

  8. Code Forces 26C Dijkstra?

    C. Dijkstra? time limit per test 1 second memory limit per test 64 megabytes input standard input ou ...

  9. 浏览器加载不上css,样式走丢

    来自:http://www.cnblogs.com/crizygo/p/5466444.html 问题描述:使用eclipse修改样式文件,浏览器的页面一时显示一时不显示,最后直接没有加载最新的css ...

  10. 使用MFC做D3D的框架

    转载请注明出处http://www.cnblogs.com/CAION/p/3192111.html (程序运行时是和其他程序挺像 = =,但我保证这是原创的) 1.将D3D的初始化,渲染等等一些行为 ...