首頁 >後端開發 >C#.Net教程 >C# IEnumerable和IEnumerator介面的簡單介紹

C# IEnumerable和IEnumerator介面的簡單介紹

黄舟
黄舟原創
2017-03-21 11:22:485420瀏覽

本文主要介紹了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,繼承泛型IEnumerable8742468051c85b06f0a0af9e3e506b5c介面即可。

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是簡化了遍歷的語法糖而已。

以上是C# IEnumerable和IEnumerator介面的簡單介紹的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn