天天看点

C# IEnumerable、IEnumerator和yield关键字详解

本文章仅为个人学习总结,如有错误请指正。

可枚举类型就是可以通过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则结束迭代

继续阅读