本文章仅为个人学习总结,如有错误请指正。
可枚举类型就是可以通过foreach循环进行运算的,支持GetEnumerator方法的(可以有参数)的类,那么这个类可以称为可枚举类型
本文章具有以下知识点
1.实现IEnumerator和IEnumerable接口。
2.yield关键字
3.foreach步骤。
会由一篇文章讲解List是如何实现foreach循环的和简单的List内部实现代码。
一、IEnumerable和IEnumerator
1.IEnumerable
该枚举数支持在非泛型集合上进行简单迭
1.1IEnumerator GetEnumerator()
返回一个循环访问集合的枚举数,可用于循环访问集合的 System.Collections.IEnumerator 对象,这个对象具体实现如何进行迭代(遍历)。
2.IEnumerator:
支持对非泛型集合的简单迭代。
2.1object Current { get; }
获取集合中的当前元素。
2.2 bool MoveNext();
将枚举数推进到集合的下一个元素。
如果枚举数成功地推进到下一个元素,则为 true;如果枚举数越过集合的结尾,则为 false。
2.3void Reset();
将枚举数设置为其初始位置,该位置位于集合中第一个元素之前。
以上是微软给出的解释,我的理解
IEnumerable
实现这个接口的类型支持某个类型的集合或数组的迭代。具体如何迭代由实现IEnumerator接口的类来确定
IEnumerator
确定指定的类(该类保证了某个类的集合具有迭代功能)如何进行迭代。
简单的说:B类型保证A类型的集合具有迭代功能,C类型去确定B类型如何进行简单的迭代(迭代返回的对象实际为A类型)
微软给出的Demo:
Person是A类型。
People是B类型,实现接口IEnumerable,支持对Person集合上进行简单迭代.
PeopleEnum是C类型,实现接口IEnumerator,实现Person集合的类型People是如何进行迭代的。
public class Person
{
public Person(string fName, string lName)
{
this.firstName = fName;
this.lastName = lName;
}
public string firstName;
public string lastName;
}
public class People : IEnumerable
{
private Person[] _people;
public People(Person[] pArray)
{
_people = new Person[pArray.Length];
for (int i = ; i < pArray.Length; i++)
{
_people[i] = pArray[i];
}
}
public IEnumerator GetEnumerator()
{
return new PeopleEnum(_people);
}
}
public class PeopleEnum : IEnumerator
{
public Person[] _people;
// Enumerators are positioned before the first element
// until the first MoveNext() call.
int position = -;
public PeopleEnum(Person[] list)
{
_people = list;
}
public bool MoveNext()
{
position++;
return (position < _people.Length);
}
public void Reset()
{
position = -;
}
public object Current
{
get
{
try
{
return _people[position];
}
catch (IndexOutOfRangeException)
{
throw new InvalidOperationException();
}
}
}
}
yield关键字,两种迭代器
yield关键字是一个语法糖,可以省略创建实现IEnumerator接口的类。
在创建可枚举类型(也就是实现了IEnumerable)接口的类型,还需要创建一个实现IEnumerator的类型,来进行确定如何实现迭代过程。
1.构建迭代器
迭代器就是一个类型的成员方法,方法名称必须为GetEnumerator(),返回值必须为IEnumerator。
但是不需要实现去创建实现IEumerator接口的类型去确定如何实现三个成员。
使用yield关键字后return 指定的对象后,当前位置会进行保存下来,下次调用的时候会从这个位置开始执行。
说白了,C#编译器会在编译后生成一个密封的类,该类实现了IEnumerator接口,所以是语法糖。
public class Person
{
public Person(string fName, string lName)
{
this.firstName = fName;
this.lastName = lName;
}
public string firstName;
public string lastName;
}
public class People
{
public People(Person[] persons)
{
_persons = persons;
}
private Person[] _persons;
public IEnumerator GetEnumerator()
{
for (int i = ; i < _persons.Length; i++)
{
yield return _persons[i];
}
}
}
2.构建命名迭代器
yield实际可以和任何方法一起使用,无论方法名称、参数名称,只要返回类型为IEnumerable接口就可以。
C#编译器内部会自动生成一个嵌套的密封类,这个类实现了接口IEnumerator和IEnumerable接口,说白了也就是一个List的部分功能。List集合实现迭代,也具有具体的迭代方法
public class Person
{
public Person(string fName, string lName)
{
this.firstName = fName;
this.lastName = lName;
}
public string firstName;
public string lastName;
}
public class People
{
private Person[] _people;
public People(Person[] pArray)
{
_people = new Person[pArray.Length];
for (int i = ; i < pArray.Length; i++)
{
_people[i] = pArray[i];
}
}
public IEnumerable GetPerson()
{
foreach (var item in _people)
{
yield return item;
}
}
}
foreach的执行步骤
foreach中var 实际是类型推断,使用使用了实际的类型,如Person那么发生理氏转换
1.执行到People后会内部调用GetEnumerator()返回,只调用一次
2.执行in,实际就是调用MoveNext,判断下一个集合元素是否存在
3.如果为true则通过属性Current获取迭代后的元素。
4.如果为false则结束迭代