当前位置: 首页 > news >正文

C#中的枚举器和迭代器

目录

一、可枚举类型和枚举器

1. 枚举器

2. 可枚举类

3. 使用 IEnumerable 和 IEnumerator 案例

4. 泛型枚举接口

二、迭代器

1. 使用迭代器创建枚举器

2. 使用迭代器创建可枚举类

3. 常见的迭代器模式

4. 产生多个枚举类型

 5. 将迭代器作为属性

6. 迭代器的实质


一、可枚举类型和枚举器

数组为什么可以被foreach语句处理?原因是数组按需提供了一个枚举器对象。

  1. foreach结构设计用来和可枚举类型一起使用。
  2. 可枚举类型可以通过GetEnumerator方法获取对象的枚举器。
  3. 从枚举器中请求每一项并且把它作为迭代变量,该变量是只读的。

1. 枚举器

枚举器需要实现IEnumerator接口,其中包含三个函数成员:Current、MoveNext、Reset。

因为数组是可枚举类型,所以我们可以通过GetEnumerator方法获取其枚举器对象,来模拟foreach遍历,通过模拟我们可以初步认识枚举器接口中的这三个方法。

    internal class Program {static void Main(string[] args){int[] ints = { 5, 6, 7 };IEnumerator enumerator = ints.GetEnumerator();//获取数组的枚举器对象while (enumerator.MoveNext()) //MoveNext方法移动指向数组中下标的指针位置,并判断是否在数组长度范围内,返回一个bool值{int current = (int)enumerator.Current;//获取当前指针指向的元素Console.WriteLine(current);}bool beforeReset = enumerator.MoveNext();Console.WriteLine("位置重置为原始位置之前,enumerator.MoveNext()方法返回:{0}", beforeReset);enumerator.Reset();//将位置重置为原始状态//当位置没有重置时,enumerator.MoveNext()返回fasle,所以不会进入这个while循环,无法遍历while (enumerator.MoveNext()) {int current = (int)enumerator.Current;Console.WriteLine(current);}}}

2. 可枚举类

前面说过数组是可枚举类型,其原因在于数组实现了IEnumerable接口。可枚举类就是指实现了IEnumerable接口的类。 

综上,可枚举类型是实现了IEnumerable接口的类,实现了IEnumerable接口中的GetEnumerator方法返回一个枚举器,枚举器是实现了 IEnumerator接口的类,通过实现IEnumerator接口三个方法来访问元素。理解图如下:

3. 使用 IEnumerable 和 IEnumerator 案例

    //枚举器class ColorEnumerator : IEnumerator{string[] _colors;int _position = -1;public ColorEnumerator(string[] colors) {_colors = new string[colors.Length];Array.Copy(colors,_colors,colors.Length);}public object Current {get{if (_position == -1)throw new InvalidOperationException();if (_position >= _colors.Length)throw new InvalidOperationException();return _colors[_position];}}public bool MoveNext(){if (_position < _colors.Length - 1){_position++;return true;}elsereturn false;            }public void Reset(){_position = -1;}}//可枚举类型class Spectrum : IEnumerable{string[] colors = { "赤", "橙", "黄", "绿", "青", "蓝", "紫" };public IEnumerator GetEnumerator(){return new ColorEnumerator(colors);}}//测试internal class Program {static void Main(string[] args){Spectrum colors = new Spectrum();foreach (string color in colors) Console.WriteLine(color);}}

4. 泛型枚举接口

  1. 泛型枚举接口的使用和非泛型枚举接口的使用是差不多的
  2. 非泛型枚举接口的实现不是类型安全的,它们返回object类型的引用,必须强转位实际类型
  3. 泛型枚举接口的实现是类型安全的,它们返回的是实际类型的对象,而非object基类的引用
  4. 应该尽量使用泛型枚举接口

二、迭代器

  1. 迭代器可以代替我们手动编码的可枚举类和枚举器。
  2. 迭代器块是有一个或多个yield语句的代码块。
  3. 迭代器块描述了希望编译器为我们创建的枚举器类的行为。
  4. 迭代器块可以是方法主体、访问器主体或运算符主体。
  5. 迭代器块中有两个特殊语句:

                i. yield return语句指定了序列中返回的下一项

                ii.yield break语句指定在序列中没有其他项

               

1. 使用迭代器创建枚举器

代码示例:

    class MyClass {public IEnumerator<string> GetEnumerator() //返回枚举器{return BlackAndWhite();}public IEnumerator<string> BlackAndWhite() //迭代器块{yield return "赤";yield return "橙";yield return "黄";yield return "绿";yield return "青";yield return "蓝";yield return "紫";}}    //测试internal class MyTest{static void Main(string[] args){MyClass myClass = new MyClass();foreach (string element in myClass)Console.WriteLine(element);}}

代码图解:

2. 使用迭代器创建可枚举类

代码示例:

    class MyClass {public IEnumerator<string> GetEnumerator() //返回枚举器{return BlackAndWhite().GetEnumerator();//返回从可枚举类获取的枚举器}public IEnumerable<string> BlackAndWhite() //返回可枚举类{yield return "赤";yield return "橙";yield return "黄";yield return "绿";yield return "青";yield return "蓝";yield return "紫";}}    //测试internal class MyTest{static void Main(string[] args){MyClass myClass = new MyClass();foreach (string element in myClass)//让类本身可枚举Console.WriteLine(element);Console.WriteLine("--------");foreach (string element in myClass.BlackAndWhite())//调用返回可枚举类的方法Console.WriteLine(element);}}

代码图解:

3. 常见的迭代器模式

通过前面的代码案例,创建迭代器可以用来产生可枚举类型和枚举器。总结如下:

  1. 如果我们创建返回枚举器的迭代器时,必须实现GetEnumerator方法来让类可枚举。
  2. 如果我们创建返回可枚举类型的迭代器时,我们有两种选择:

                选择1:实现GetEnumerator让类本身可枚举

                选择2:不实现GetEnumerator让类本身不可枚举,但仍可使用由迭代器产生的可枚举类

4. 产生多个枚举类型

可以在同一个类中创建多个迭代器来产生多个枚举类型。

    //注意:该类中没有实现GetEnumerator方法,所以该类本身不可以被枚举,但可以通过迭代器返回的枚举类型进行遍历class MyClass {string[] colors = { "赤", "橙", "黄", "绿", "青", "蓝", "紫" };public IEnumerable<string> PrintOut() //迭代器返回可枚举类型{for (int i = 0; i < colors.Length; i++)yield return colors[i];}public IEnumerable<string> ReversePrintOut() //迭代器返回可枚举类型{for(int i = colors.Length-1; i >= 0; i--)yield return colors[i];}}    //测试internal class MyTest{static void Main(string[] args){MyClass myClass = new MyClass();foreach (string color in myClass.PrintOut())Console.WriteLine(color);Console.WriteLine("--------");foreach (string color in myClass.ReversePrintOut())Console.WriteLine(color);}}

 5. 将迭代器作为属性

可以将迭代器作为属性。代码示例如下:

    class Colors {bool chooseEnumerator;string[] colors = { "赤", "橙", "黄", "绿", "青", "蓝", "紫" };public Colors(bool b) {chooseEnumerator = b;}//根据创建类对象时传入的布尔值控制返回不同的枚举器public IEnumerator<string> GetEnumerator() {return chooseEnumerator ? PrintOut : ReversePrintOut;}public IEnumerator<string> PrintOut //迭代器放置在属性get访问器中{get {for (int i = 0; i < colors.Length; i++)yield return colors[i];}}public IEnumerator<string> ReversePrintOut //迭代器放置在属性get访问器中{get {for (int i = colors.Length - 1; i >= 0; i--)yield return colors[i];}}}    //测试internal class MyTest{static void Main(string[] args){Colors colors = new Colors(true);foreach (string color in colors)Console.WriteLine(color);Console.WriteLine("--------");Colors reColors = new Colors(false);foreach (string color in reColors)Console.WriteLine(color);}}

6. 迭代器的实质

  1. 迭代器需要using指令引入System.Collections.Generic的命名空间。
  2. 在编译器生成的枚举器中,Reset方法没有实现,调用会抛异常。
  • 由编译器生成的枚举器是包含四个状态的状态机。
  • Before:首次调用MoveNext的初始状态。
  • Running:调用MoveNext后进入这个状态。在这个状态中,枚举器检查并设置下一项的位置,在遇到yield return、yield break或在迭代器体结束时,退出状态。
  • Suspended:状态机等待下次调用MoveNext的状态。
  • After:没有更多项可以枚举。

 

(注:本章学习总结自《C#图解教程》)

http://www.lryc.cn/news/48342.html

相关文章:

  • 中山大学人工智能学院——考研上岸经验贴
  • ThreeJS-圣诞节表白3D贺卡(三十)
  • 040:cesium加载World Terrain地形图
  • 逻辑运算和位移指令
  • 大家现在都去做Linux运维了吗?
  • Webpack的编译流程是怎么样的?webpack是如何工作的?
  • 【ZOJ 1151】Word Reversal 题解(字符串+模拟)
  • Dart语言操作符?和!的用法
  • 聚类 kmeans | 机器学习
  • 求职咨询Job Information
  • 怎么去除pdf文件的水印?好用软件说明
  • 1-ELK+ Elasticsearch+head+kibana、企业内部日志分析系统
  • ctfshow愚人杯web复现
  • 商品推荐Promoting Products
  • 整懵了,蚂蚁金服4面成功拿下测开offer,涨薪6k,突然觉得跳槽也不是那么难
  • 《扬帆优配》个人养老金投资最新成绩出炉 七成养老FOF跑输基准
  • 用Qt编写STM32烧录软件(ISP模式)代码
  • Excel技能之美观排版
  • 兆芯最新X86 CPU曝光:性能与英特尔/AMD相比,没落后10年
  • 【Go自学】一文搞懂Go的strconv模块
  • SpringBoot整合Admin服务监控(图文详细)
  • 设计模式-结构型模式-组合模式
  • VScode开发工具总结
  • opencv 解码视频流,c++ 代码写法
  • Android 12.0 修改wifi信号强度
  • Linux——容器简介
  • CMOS图像传感器——pipeline像素控制
  • AI工具(ChatGPT)常用指令,持续更新...
  • 36--Django-项目实战-全栈开发-基于django+drf+vue+elementUI企业级项目开发流程-前台项目准备
  • 游戏算法-游戏AI行为树,python实现