初學C#的時候,總是被IEnumerable、IEnumerator、ICollection等這樣的接口弄的糊里糊塗,我以爲有必要切底的弄清楚IEnumerable和IEnumerator的本質。數組
下面咱們先看IEnumerable和IEnumerator兩個接口的語法定義。其實IEnumerable接口是很是的簡單,只包含一個抽象的方法GetEnumerator(),它返回一個可用於循環訪問集合的IEnumerator對象。IEnumerator對象有什麼呢?它是一個真正的集合訪問器,沒有它,就不能使用foreach語句遍歷集合或數組,由於只有IEnumerator對象才能訪問集合中的項,假如連集合中的項都訪問不了,那麼進行集合的循環遍歷是不可能的事情了。那麼讓咱們看看IEnumerator接口有定義了什麼東西。看下圖咱們知道IEnumerator接口定義了一個Current屬性,MoveNext和Reset兩個方法,這是多麼的簡約。既然IEnumerator對象時一個訪問器,那至少應該有一個Current屬性,來獲取當前集合中的項吧。安全
MoveNext方法只是將遊標的內部位置向前移動(就是移到一下個元素而已),要想進行循環遍歷,不向前移動一下怎麼行呢?函數


詳細講解:測試
說到IEnumerable老是會和IEnumerator、foreach聯繫在一塊兒。this
C# 支持關鍵字foreach,容許咱們遍歷任何數組類型的內容:spa
//遍歷數組的項.net
int[] myArrayOfInts = {10,20,30,40};對象
foreach(int i in my myArrayOfInts)blog
{繼承
Console.WirteLine(i);
}
雖然看上去只有數組纔可使用這個結構,其實任何支持GetEnumerator()方法的類型均可以經過foreach結構進行運算。
- public class Garage
- {
- Car[] carArray = new Car[4];
-
-
- public Garage()
- {
-
- carArray[0] = new Car("Rusty", 30);
- carArray[1] = new Car("Clunker", 50);
- carArray[2] = new Car("Zippy", 30);
- carArray[3] = new Car("Fred", 45);
- }
- }
理想狀況下,與數據值數組同樣,使用foreach構造迭代Garage對象中的每個子項比較方便:
- lass Program
- {
- static void Main(string[] args)
- {
- Console.WriteLine("*********Fun with IEnumberable/IEnumerator************\n");
- Garage carLot = new Garage();
-
-
- foreach (Car c in carLot)
- {
- Console.WriteLine("{0} is going {1} MPH", c.CarName, c.CurrentSpeed);
- }
-
- Console.ReadLine();
- }
- }
讓人沮喪的是,編譯器通知咱們Garage類沒有實現名爲GetEnumerator()的方法(顯然用foreach遍歷Garage對象是不可能的事情,由於Garage類沒有實現GetEnumerator()方法,Garage對象就不可能返回一個IEnumerator對象,沒有IEnumerator對象,就不可能調用方法MoveNext(),調用不了MoveNext,就不可能循環的了)。這個方法是有隱藏在System.collections命名空間中的IEnumerable接口定義的。(特別注意,其實咱們循環遍歷的都是對象而不是類,只是這個對象是一個集合對象)
支持這種行爲的類或結構其實是宣告它們向調用者公開所包含的子項:
//這個接口告知調方對象的子項能夠枚舉
public interface IEnumerable
{
IEnumerator GetEnumerator();
}
能夠看到,GetEnumerator方法返回對另外一個接口System.Collections.IEnumerator的引用。這個接口提供了基礎設施,調用方能夠用來移動IEnumerable兼容容器包含的內部對象。
//這個接口容許調用方獲取一個容器的子項
public interface IEnumerator
{
bool MoveNext(); //將遊標的內部位置向前移動
object Current{get;} //獲取當前的項(只讀屬性)
void Reset(); //將遊標重置到第一個成員前面
}
因此,要想Garage類也可使用foreach遍歷其中的項,那咱們就要修改Garage類型使之支持這些接口,能夠手工實現每個方法,不過這得花費很多功夫。雖然本身開發GetEnumerator()、MoveNext()、Current和Reset()也沒有問題,但有一個更簡單的辦法。由於System.Array類型和其餘許多類型(如List)已經實現了IEnumerable和IEnumerator接口,你能夠簡單委託請求到System.Array,以下所示:
- namespace MyCarIEnumerator
- {
- public class Garage:IEnumerable
- {
- Car[] carArray = new Car[4];
-
-
- public Garage()
- {
- carArray[0] = new Car("Rusty", 30);
- carArray[1] = new Car("Clunker", 50);
- carArray[2] = new Car("Zippy", 30);
- carArray[3] = new Car("Fred", 45);
- }
- public IEnumerator GetEnumerator()
- {
- return this.carArray.GetEnumerator();
- }
- }
- }
- namespace MyCarIEnumerator
- {
- class Program
- {
- static void Main(string[] args)
- {
- Console.WriteLine("*********Fun with IEnumberable/IEnumerator************\n");
- Garage carLot = new Garage();
-
-
- foreach (Car c in carLot)
- {
- Console.WriteLine("{0} is going {1} MPH", c.CarName, c.CurrentSpeed);
- }
-
- Console.WriteLine("GetEnumerator被定義爲公開的,對象用戶能夠與IEnumerator類型交互,下面的結果與上面是一致的");
-
- IEnumerator i = carLot.GetEnumerator();
- while (i.MoveNext())
- {
- Car myCar = (Car)i.Current;
- Console.WriteLine("{0} is going {1} MPH", myCar.CarName, myCar.CurrentSpeed);
- }
- Console.ReadLine();
- }
- }
- }
下面咱們來看看手工實現IEnumberable接口和IEnumerator接口中的方法:
- namespace ForeachTestCase
- {
-
- class ForeachTest:IEnumerable {
- private string[] elements;
- private int ctr = 0;
-
-
-
-
-
- ForeachTest(params string[] initialStrings)
- {
-
- elements = new String[8];
-
- foreach (string s in initialStrings)
- {
- elements[ctr++] = s;
- }
- }
-
-
-
-
-
-
- ForeachTest(string initialStrings, char[] delimiters)
- {
- elements = initialStrings.Split(delimiters);
- }
-
-
- public IEnumerator GetEnumerator()
- {
- return new ForeachTestEnumerator(this);
- }
-
- private class ForeachTestEnumerator : IEnumerator
- {
- private int position = -1;
- private ForeachTest t;
- public ForeachTestEnumerator(ForeachTest t)
- {
- this.t = t;
- }
-
- #region 實現接口
-
- public object Current
- {
- get
- {
- return t.elements[position];
- }
- }
-
- public bool MoveNext()
- {
- if (position < t.elements.Length - 1)
- {
- position++;
- return true;
- }
- else
- {
- return false;
- }
- }
-
- public void Reset()
- {
- position = -1;
- }
-
- #endregion
- }
- static void Main(string[] args)
- {
-
- ForeachTest f = new ForeachTest("This", "is", "a", "sample", "sentence.");
- foreach (string item in f)
- {
- System.Console.WriteLine(item);
- }
- Console.ReadKey();
- }
- }
- }
IEnumerable<T>接口
實現了IEnmerable<T>接口的集合,是強類型的。它爲子對象的迭代提供類型更加安全的方式。
- public class ListBoxTest:IEnumerable<String>
- {
- private string[] strings;
- private int ctr = 0;
-
- #region IEnumerable<string> 成員
-
- public IEnumerator<string> GetEnumerator()
- {
- foreach (string s in strings)
- {
- yield return s;
- }
- }
-
- #endregion
-
- #region IEnumerable 成員
-
- System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
- {
- return GetEnumerator();
- }
-
- #endregion
-
-
- public ListBoxTest(params string[] initialStrings)
- {
-
- strings = new String[8];
-
- foreach (string s in initialStrings)
- {
- strings[ctr++] = s;
- }
- }
-
-
- public void Add(string theString)
- {
- strings[ctr] = theString;
- ctr++;
- }
-
-
- public string this[int index]
- {
- get {
- if (index < 0 || index >= strings.Length)
- {
-
- }
- return strings[index];
- }
- set {
- strings[index] = value;
- }
- }
-
-
- public int GetNumEntries()
- {
- return ctr;
- }
- }
- class Program
- {
- static void Main(string[] args)
- {
-
- ListBoxTest lbt = new ListBoxTest("Hello", "World");
-
-
- lbt.Add("Who");
- lbt.Add("Is");
- lbt.Add("Douglas");
- lbt.Add("Adams");
-
-
- string subst = "Universe";
- lbt[1] = subst;
-
-
- foreach (string s in lbt)
- {
- Console.WriteLine("Value:{0}", s);
- }
- Console.ReadKey();
- }
- }
綜上所述,一個類型是否支持foreach遍歷,必須知足下面條件:
方案1:讓這個類實現IEnumerable接口
方案2:這個類有一個public的GetEnumerator的實例方法,而且返回類型中有public 的bool MoveNext()實例方法和public的Current實例屬性。