C# IEnumerable和IEnumerator接口淺析。本站提示廣大學習愛好者:(C# IEnumerable和IEnumerator接口淺析)文章只能為提供參考,不一定能成為您想要的結果。以下是C# IEnumerable和IEnumerator接口淺析正文
溫故而知新,可以為師矣,有空經常復習一下基礎知識是有必要的,並且能加深理解和記憶。
Foreach常用於循環訪問集合,對實現IEnumerable的接口的容器進行遍歷,IEnumerable和IEnumerator接口我有時候也有點迷糊,按官方的解釋,IEnumerable是枚舉器接口,IEnumerator是迭代器接口,從字面意思來看相差不大,逐一分析一下。
IEnumerable接口
public interface IEnumerable
{
IEnumerator GetEnumerator();
}
繼承IEnumerable接口的類需實現暴露出來的GetEnumerator()方法,並返回一個IEnumerator接口對象,看來真正做事的是IEnumerator,F12看一下IEnumerator又有什麼鬼東西。
IEnumerator接口
public interface IEnumerator
{
object Current { get; }
bool MoveNext();
void Reset();
}
IEnumerator接口有三個東東,一個屬性Current,返回當前集合中的元素,方法MoveNext()移動到下一個,遍歷不都是向後遍歷的嘛,Reset(),字面意思重置,這個容易理解。做個假設:既然IEnumerable接口返回是IEnumerator接口迭代器來實現的,那麼僅繼承IEnumerator迭代器接口能不能實現一個自定義容器?
定義一個Phone類
public class Phone
{
public string Name;
public Phone(string name)
{
this.Name = name;
}
}
定義一個名為MyEnumerator迭代器,並現實它接口IEnumerator
public class MyEnumerator : IEnumerator
{
Phone[] p;
int idx = -1;
public MyEnumerator(Phone[] t)
{
p = t;
}
public object Current
{
get
{
if (idx == -1)
return new IndexOutOfRangeException();
return p[idx];
}
}
public bool MoveNext()
{
idx++;
return p.Length > idx;
}
public void Reset()
{
idx = -1;
}
}
class Program
{
static void Main(string[] args)
{
show("-----------IEnumerator------------");
Phone[] phones = new Phone[] { new Phone("iPhone 7s"), new Phone("iPhone 6s"), new Phone("iPhone 5s") };
MyEnumerator enumerator = new MyEnumerator(phones);
while (enumerator.MoveNext())
{
Phone p = enumerator.Current as Phone;
show(p.Name);
}
Console.ReadKey();
}
static void show(string i)
{
Console.WriteLine(i);
}
}
結果顯示:

果然不出所料,真正做事情的是IEnumerator接口,即可循環訪問自定義的一個容器,不過,初衷是想用Foreach來做循環訪問、遍歷的。那好,那就只能顯示IEnumerable接口來做。稍稍改造一下Phone類:
public class Phone : IEnumerable
{
public string Name ;
public Phone(string name)
{
this.Name = name;
}
Phone[] p;
public Phone(Phone[] t)
{
p = t;
}
public IEnumerator GetEnumerator()
{
return new MyEnumerator(p);
}
}
static void Main(string[] args)
{
show("-----------IEnumerator------------");
Phone[] phones = new Phone[] { new Phone("iPhone 7s"), new Phone("iPhone 6s"), new Phone("iPhone 5s") };
MyEnumerator enumerator = new MyEnumerator(phones);
while (enumerator.MoveNext())
{
Phone p = enumerator.Current as Phone;
show(p.Name);
}
show("-----------IEnumerable------------");
Phone phoneList = new Phone(phones);
foreach (Phone p in phoneList)
{
show(p.Name);
}
Console.ReadKey();
}
結果顯示:

大功告成,再擴展成通用的容器PhonePackage,繼承泛型IEnumerable<T>接口即可。
public class PhonePackage<T> : IEnumerable<T>
{
private List<T> dataList = null;
public void Add(T t)
{
if (dataList == null)
dataList = new List<T>();
dataList.Add(t);
}
public IEnumerator<T> GetEnumerator()
{
foreach (T t in dataList)
{
yield return t;
}
}
IEnumerator IEnumerable.GetEnumerator()
{
foreach (T t in dataList)
{
yield return t;
}
}
}
static void Main(string[] args)
{
show("-----------IEnumerator------------");
Phone[] phones = new Phone[] { new Phone("iPhone 7s"), new Phone("iPhone 6s"), new Phone("iPhone 5s") };
MyEnumerator enumerator = new MyEnumerator(phones);
while (enumerator.MoveNext())
{
Phone p = enumerator.Current as Phone;
show(p.Name);
}
show("-----------IEnumerable------------");
Phone phoneList = new Phone(phones);
foreach (Phone p in phoneList)
{
show(p.Name);
}
show("-----------IEnumerable<T>------------");
PhonePackage<Phone> phonePackage = new PhonePackage<Phone>();
phonePackage.Add(new Phone("iPhone 7s"));
phonePackage.Add(new Phone("iPhone 6s"));
phonePackage.Add(new Phone("iPhone 5s"));
foreach (Phone p in phonePackage)
{
show(p.Name);
}
Console.ReadKey();
}
static void show(string i)
{
Console.WriteLine(i);
}
結果顯示:

IEnumerator迭代器接口挺啰嗦的,yield是簡化了遍歷的語法糖而已。
以上就是本文的全部內容,希望本文的內容對大家的學習或者工作能帶來一定的幫助,同時也希望多多支持!
[db:作者簡介][db:原文翻譯及解析]