您的位置:首页 > 其它

IEnumerable 接口---实现该接口里的成员的类,可以使用foreach循环访问集合

2010-12-18 01:10 766 查看
IEnumerable 接口支持对非泛型集合的简单迭代。 实现该接口里的成员的类,可以使用foreach循环访问集合(VB为For Each)。

备注
IEnumerator 是所有非泛型枚举数的基接口。
有关此接口的泛形版本,请参见 IEnumerator<(Of <(T>)>)。
C# 语言的 foreach 语句(在 Visual Basic 中为 for each)隐藏了枚举数的复杂性。因此,建议使用 foreach,而不直接操作枚举数。
枚举数可用于读取集合中的数据,但不能用于修改基础集合。
最初,枚举数定位在集合中第一个元素前。Reset 方法还会将枚举数返回到此位置。在此位置,调用 Current 属性会引发异常。因此,在读取 Current 的值之前,必须调用 MoveNext 方法将枚举数提前到集合的第一个元素。
在调用 MoveNext 或 Reset 之前,Current 返回同一对象。MoveNext 将 Current 设置为下一个元素。
如果 MoveNext 越过集合的末尾,则枚举数将被放置在此集合中最后一个元素的后面,而且 MoveNext 返回 false。当枚举数位于此位置时,对 MoveNext 的后续调用也返回 false。如果最后一次调用 MoveNext 返回 false,则调用 Current 会引发异常。若要再次将 Current 设置为集合的第一个元素,可以调用 Reset,然后再调用 MoveNext。
只要集合保持不变,枚举数就保持有效。如果对集合进行了更改(如添加、修改或删除元素),则枚举数将失效且不可恢复,并且下一次对 MoveNext 或 Reset 的调用将引发 InvalidOperationException。如果在 MoveNext 和 Current 之间修改集合,那么即使枚举数已经无效,Current 也将返回它所设置成的元素。
枚举数没有对集合的独占访问权;因此,枚举通过集合在本质上不是一个线程安全的过程。即使一个集合已进行同步,其他线程仍可以修改该集合,这将导致枚举数引发异常。若要在枚举过程中保证线程安全,可以在整个枚举过程中锁定集合,或者捕捉由于其他线程进行的更改而引发的异常。

示例
下面的代码示例演示如何实现自定义集合的 IEnumerable 和 IEnumerator 接口。在此示例中,没有显式调用这些接口的成员,但实现了它们,以便支持使用 foreach(在 Visual Basic 中为 For Each)循环访问该集合。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

//
using System.Collections;

namespace MvcUsedKeywords
{
public class Person
{
public string firstName;
public string lastName;
public Person(string fName, string lName)
{
this.firstName = fName;
this.lastName = lName;
}
}

//实现IEnumerable接口中的方法
public class People : IEnumerable
{
private Person[] _people;
public People(Person[] pArray)
{
_people=new Person[pArray.Length];
for (int i = 0; i < pArray.Length; i++)
{
_people[i]=pArray[i];
}
}

public IEnumerator GetEnumerator()
{
return new PeopleEnum(_people);
}
}

//实现IEnumerator接口中的方法
public class PeopleEnum : IEnumerator
{
public Person[] _people;
int position = -1;

public PeopleEnum(Person[] list)
{
_people = list;
}

public bool MoveNext()
{
position++;
return (position < _people.Length);
}

public void Reset()
{
position = -1;
}

public object Current
{
get
{
try
{
return _people[position];
}
catch (IndexOutOfRangeException)
{
throw new InvalidOperationException();
}
}
}
}

class App
{
static void Main(string[] args)
{
Person[] peopleArray = new Person[3]{
new Person("John","Smith"),
new Person("Jim","Johnson"),
new Person("Sue","Rabon"),
};

//利用实现IEnumerable接口中方法的类,利用foreach遍历集合
Console.WriteLine("利用实现IEnumerable接口中方法的类:");
People peopleList = new People(peopleArray);
foreach (Person p in peopleList)
Console.WriteLine(p.firstName + " " + p.lastName);

PeopleEnum peopleEnum = new PeopleEnum(peopleArray);

//利用实现IEnumerator接口中方法的类,利用实现的方法来遍历集合
Console.WriteLine("/n利用实现IEnumerator接口中方法的类:");
Person person=null;
while (peopleEnum.MoveNext())
{
person = (Person)peopleEnum.Current;
Console.WriteLine(person.firstName + " " + person.lastName);
}

}
}
}

运行结果
利用实现IEnumerable接口中方法的类:
John Smith
Jim Johnson
Sue Rabon
利用实现IEnumerator接口中方法的类:
John Smith
Jim Johnson
Sue Rabon
请按任意键继续. . .
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐