更新记录

本文迁移自Panda666原博客,原发布时间:2021年6月28日。

一、先从可枚举类型讲起

1.1 什么是可枚举类型?



可枚举类型,可以简单的理解为:

有一个类,类中有挺多的数据,用一种统一的方式把他们列举出来。

在.NET中满足以下任意条件的都是可枚举类型:

Implements System.Collections.IEnumerable

Implements System.Collections.Generic.IEnumerable

Has a public parameterless method named GetEnumerator that returns an enumerator

这三条看起来非常的简单。第一条实现了IEnumerable接口就行了。第二条实现了IEnumerable接口就行了。第三条就更简洁了,实现了GetEnumerator方法就行了。所以可以发现,可枚举类型实现起来非常的方便。

1.2 那么问题来了,为什么需要可枚举类型?

上面不是说了吗,用一种统一的方式把数据列举出来。

那么问题来了,为什么需要用统一的方式取出数据?

因为方便取出数据,一百个类有一百个取出数据的方式,维护起来就很难受了。

所以统一数据的取出方式,是非常有必要的。

1.3 具体实现可枚举类型

说了这么多,是什么,为什么,来看看怎么用。我们先来看看,上面说到的2个接口。

System.Collections.IEnumerable接口的定义

namespace System.Collections
{
public interface IEnumerable
{
IEnumerator GetEnumerator();
}
}

System.Collections.Generic.IEnumerable接口 的定义

public interface IEnumerable<out T> : IEnumerable
{
IEnumerator<T> GetEnumerator();
}

注意点:

可以看到2个接口大同小异,都有GetEnumerator方法,但要注意的是2个方法的返回类型是不同的。

GetEnumerator从字面意思也能看出来,表示获得Enumerator。

那么什么是Enumerator?Enumerator表示枚举器。

我们的数据统一数据的取出方式全部都靠枚举器帮我们代劳,而不是直接定义在本类中,当然你也可以这样做,但不建议。接下来现在我们来定义几个可枚举类型。

继承自IEnumerable的可枚举类型

public class PandaClass1 : IEnumerable
{
public IEnumerator GetEnumerator()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
}
}

继承自IEnumerable的可枚举类型

public class PandaClass2 : IEnumerable<int>
{
public IEnumerator<int> GetEnumerator()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
}
IEnumerator IEnumerable.GetEnumerator()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
}
}

直接实现GetEnumerator方法的可枚举类型

public class PandaClass3
{
public IEnumerator GetEnumerator()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
}
}

直接实现GetEnumerator方法的可枚举类型

public class PandaClass4
{
public IEnumerator<int> GetEnumerator()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
}
}

到这里我们已经搞定可枚举类型了,怎么进行枚举?

二、有趣的枚举器(Enumerator)

2.1 那么问题来了,什么是枚举器。

Enumerator在英文中有列举出人或物的意思。所以是列举器的意思?不急,我们来对比看看程序中什么是枚举器。

在.NET中满足以下任意条件的都是枚举器:

1、Implements System.Collections.IEnumerator

2、Implements System.Collections.Generic.IEnumerator

3、Has a public parameterless method named MoveNext and property called Current

第三条中的实现MoveNext方法和Current属性倒是挺好理解的,只要实现了这2个成员就叫枚举器。

比如下面这类就叫枚举器

public class Enumerator
{
public IteratorVariableType Current { get {...} }
public bool MoveNext() {...}
}

那么第一二条中的接口是什么,不慌,我们先来看看这2个接口的定义。

IEnumerator接口的定义

namespace System.Collections
{
public interface IEnumerator
{
object Current { get; }
bool MoveNext();
void Reset();
}
}

IEnumerator接口的定义

namespace System.Collections.Generic
{
public interface IEnumerator<out T> : IEnumerator, IDisposable
{
T Current { get; }
}
}

需要注意的点:

IEnumerator接口返回的是object类型。

从接口的定义我们可以发现IEnumerator继承自IEnumerator。

所以IEnumerator是同样包含的有MoveNext()和Reset()方法的。

MoveNext()方法表示移动到下一个元素。

Reset()方法表示重置枚举器。

Current属性用于获得当前的元素值。

2.2 实现枚举器

直接继承IEnumerator接口实现枚举器

public class PandaEnumerator1 : IEnumerator
{
//内部没有具体实现,所以先抛出异常
public object Current => throw new NotImplementedException(); public bool MoveNext()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
} public void Reset()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
}
}

直接继承IEnumerator接口实现枚举器

public class PandaEnumerator2 : IEnumerator<int>
{
//内部没有具体实现,所以先抛出异常
public int Current => throw new NotImplementedException(); //内部没有具体实现,所以先抛出异常
object IEnumerator.Current => throw new NotImplementedException(); //用于释放枚举器使用的资源
public void Dispose()
{
throw new NotImplementedException();
} public bool MoveNext()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
} public void Reset()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
}
}

直接实现要求的方法实现枚举器

public class PandaEnumerator3
{
//内部没有具体实现,所以先抛出异常
public object Current => throw new NotImplementedException(); public bool MoveNext()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
} public void Reset()
{
//内部没有具体实现,所以先抛出异常
throw new NotImplementedException();
}
}

三、组合枚举类型和枚举器

那么问题来了,枚举类型和枚举器都搞定了。怎么把他们组合起来?怎么用来?

我们这里用一个实例来进行演示。我们先定义一个学生类来表示要枚举的数据。

/// <summary>
/// 学生数据
/// </summary>
public class Student
{
/// <summary>
/// 编号
/// </summary>
public string Id { get; set; } /// <summary>
/// 姓名
/// </summary>
public string Name { get; set; } /// <summary>
/// 年龄
/// </summary>
public int Age { get; set; }
}

然后我们定义枚举器

/// <summary>
/// 枚举器
/// </summary>
public class StudentIEnumerator : IEnumerator<Student>
{
/// <summary>
/// 用于遍历的学生信息
/// </summary>
private List<Student> InnerStudents { get; set; } /// <summary>
/// 当前的位置
/// </summary>
private int CurrentPosition { get; set; } public StudentIEnumerator(List<Student> students)
{
this.InnerStudents = students;
this.CurrentPosition = -1;
} //获得当前的元素
public Student Current => this.InnerStudents[this.CurrentPosition]; //如果无需兼容老代码,可以无需修改
object IEnumerator.Current => throw new NotImplementedException(); public void Dispose()
{ } public bool MoveNext()
{
if(this.CurrentPosition < this.InnerStudents.Count-1)
{
++this.CurrentPosition;
return true;
} return false;
} public void Reset()
{
this.CurrentPosition = -1;
}
}

最后定义可枚举类型

/// <summary>
/// 可枚举类型
/// </summary>
public class StudentCollecion : IEnumerable<Student>
{
/// <summary>
/// 学生数据
/// </summary>
public List<Student> Students { get; set; } public StudentCollecion()
{
//初始化List
this.Students = new List<Student>();
//虚构用于测试的学生信息
this.Students.Add(new Student() {
Id = "666",
Name = "Panda",
Age = 18
}); this.Students.Add(new Student()
{
Id = "888",
Name = "Donkey",
Age = 18
}); this.Students.Add(new Student()
{
Id = "999",
Name = "Dog",
Age = 18
}); this.Students.Add(new Student()
{
Id = "777",
Name = "Pig",
Age = 18
}); this.Students.Add(new Student()
{
Id = "333",
Name = "Monkey",
Age = 18
}); } /// <summary>
/// 获得枚举器
/// 返回我们自定义的配套枚举器
/// </summary>
/// <returns></returns>
public IEnumerator<Student> GetEnumerator()
{
return new StudentIEnumerator(this.Students);
} //如果无需兼容老代码可以不用修改
IEnumerator IEnumerable.GetEnumerator()
{
throw new NotImplementedException();
}
}

四、使用可枚举类型

那么问题来了,怎么使用可枚举类型?

方式一:使用foreach。这是最常用的方式。

StudentCollecion students = new StudentCollecion();

foreach (var student in students)
{
Console.WriteLine(student.Id);
}

注意:foreach只是方式二的语法糖。如果枚举器实现了IDisposable接口,foreach会自动调用dispose方法。

方式二:直接使用迭代器

StudentCollecion students = new StudentCollecion();
//获得实例的枚举器
IEnumerator<Student> enumerator = students.GetEnumerator(); //手动操作
while (enumerator.MoveNext())
{
//获得元素值
Console.WriteLine(enumerator.Current.Name);
} //重置一下枚举器
enumerator.Reset();

五、总结

实例完整源代码

using System;
using System.Collections;
using System.Collections.Generic;
namespace Panda666comTest
{
/// <summary>
/// 枚举器
/// </summary>
public class StudentIEnumerator : IEnumerator<Student>
{
/// <summary>
/// 用于遍历的学生信息
/// </summary>
private List<Student> InnerStudents { get; set; } /// <summary>
/// 当前的位置
/// </summary>
private int CurrentPosition { get; set; } public StudentIEnumerator(List<Student> students)
{
this.InnerStudents = students;
this.CurrentPosition = -1;
} //获得当前的元素
public Student Current => this.InnerStudents[this.CurrentPosition]; //如果无需兼容老代码,可以无需修改
object IEnumerator.Current => throw new NotImplementedException(); public void Dispose()
{ } public bool MoveNext()
{
if (this.CurrentPosition < this.InnerStudents.Count - 1)
{
++this.CurrentPosition;
return true;
}
return false;
} public void Reset()
{
this.CurrentPosition = -1;
}
} /// <summary>
/// 学生数据
/// </summary>
public class Student
{
/// <summary>
/// 编号
/// </summary>
public string Id { get; set; } /// <summary>
/// 姓名
/// </summary>
public string Name { get; set; } /// <summary>
/// 年龄
/// </summary>
public int Age { get; set; }
} /// <summary>
/// 可枚举类型
/// </summary>
public class StudentCollecion : IEnumerable<Student>
{
/// <summary>
/// 学生数据
/// </summary>
public List<Student> Students { get; set; }
public StudentCollecion()
{
//初始化List
this.Students = new List<Student>(); //虚构用于测试的学生信息
this.Students.Add(new Student()
{
Id = "666",
Name = "Panda",
Age = 18
}); this.Students.Add(new Student()
{
Id = "888",
Name = "Donkey",
Age = 18
}); this.Students.Add(new Student()
{
Id = "999",
Name = "Dog",
Age = 18
}); this.Students.Add(new Student()
{
Id = "777",
Name = "Pig",
Age = 18
}); this.Students.Add(new Student()
{
Id = "333",
Name = "Monkey",
Age = 18
});
} /// <summary>
/// 获得枚举器
/// 返回我们自定义的配套枚举器
/// </summary>
/// <returns></returns>
public IEnumerator<Student> GetEnumerator()
{
return new StudentIEnumerator(this.Students);
} //如果无需兼容老代码可以不用修改
IEnumerator IEnumerable.GetEnumerator()
{
throw new NotImplementedException();
}
} class Program
{
static void Main(string[] args)
{
//实例化可枚举类型
StudentCollecion students = new StudentCollecion(); //方式一:使用foreach来遍历可枚举类型
foreach (var student in students)
{
Console.WriteLine(student.Id);
} //方式二:使用手动调用枚举器
//获得实例的枚举器
IEnumerator<Student> enumerator = students.GetEnumerator(); //手动操作
while (enumerator.MoveNext())
{
//获得元素值
Console.WriteLine(enumerator.Current.Name);
} //重置一下枚举器
enumerator.Reset();
enumerator.Dispose(); //wait
Console.WriteLine("Success");
Console.ReadKey();
}
}
}

C#中的枚举器的更多相关文章

  1. ruby中迭代器枚举器的理解

    参考<ruby编程语言>5.3迭代器和可枚举对象 迭代器一个迭代器是一个方法,这个方法里面有yield语句,这个方法里的yield语句,与传递给这个方法的块进行数据传输 yield将数据传 ...

  2. C#中的枚举器(转)

    术语表 Iterator:枚举器(迭代器) 如果你正在创建一个表现和行为都类似于集合的类,允许类的用户使用foreach语句对集合中的成员进行枚举将会是很方便的.这在C# 2.0中比 C# 1.1更容 ...

  3. C#图解教程 第十八章 枚举器和迭代器

    枚举器和迭代器 枚举器和可枚举类型 foreach语句 IEnumerator接口 使用IEnumerable和IEnumerator的示例 泛型枚举接口迭代器 迭代器块使用迭代器来创建枚举器使用迭代 ...

  4. JAVA中的数据结构——集合类(序):枚举器、拷贝、集合类的排序

    枚举器与数据操作 1)枚举器为我们提供了访问集合的方法,而且解决了访问对象的“数据类型不确定”的难题.这是面向对象“多态”思想的应用.其实是通过抽象不同集合对象的共同代码,将相同的功能代码封装到了枚举 ...

  5. C#中的foreach语句与枚举器接口(IEnumerator)及其泛型 相关问题

    这个问题从<C#高级编程>数组一节中的foreach语句(6.7.2)发现的. 因为示例代码与之前的章节连贯,所以我修改了一下,把自定义类型改为了int int[] bs = { 2, 3 ...

  6. C#2.0中使用yield关键字简化枚举器的实现

    我们知道要使用foreach语句从客户端代码中调用迭代器,必需实现IEnumerable接口来公开枚举器,IEnumerable是用来公开枚举器的,它并不实现枚举器,要实现枚举器必需实现IEnumer ...

  7. Python 中的枚举类型~转

    Python 中的枚举类型 摘要: 枚举类型可以看作是一种标签或是一系列常量的集合,通常用于表示某些特定的有限集合,例如星期.月份.状态等. 枚举类型可以看作是一种标签或是一系列常量的集合,通常用于表 ...

  8. for..in遍历,枚举器

    #pragma mark ------------for循环遍历集合中的元素------ //创建一个数组,包含5个字符串对象,倒序取出数组中的所有元素,并存储到另一可变数组中 NSArray *ar ...

  9. ruby迭代器枚举器

    迭代器一个迭代器是一个方法,这个方法里面有yield语句,使用了yield的方法叫做迭代器,迭代器并非一定要迭代,与传递给这个方法的块进行数据传输 yield将数据传给代码快,代码块再把数据传输给yi ...

随机推荐

  1. Redis快速入门到精通

    Redis Redis是一个开源的使用ANSI C语言编写.支持网络. 可基于内存亦可持久化的日志型.Key-Value型 NoSQL数据库,并提供多种语言的API.从2010年3 月15日起,Red ...

  2. SpringMVC-获得Restful风格的参数

    使用@PathVariable注解:接收请求路径中占位符的值 @RequestMapping("/report18/{username}") @ResponseBody publi ...

  3. Java对象和多态

    Java对象和多态 (面向对象) 面向对象基础 面向对象程序设计(Object Oriented Programming) 对象基于类创建,类相当于一个模板,对象就是根据模板创建出来的实体(就像做月饼 ...

  4. sourceCRT设置全局字符集为utf-8

    以前刚打开服务器crt字符集都会默认是default模式,搞得每次都要手动设置成UTF-8. 烦躁. 将CRT全局字符集设置成UTF-8格式方法: 设置窗口不会断掉: 即每100s发送一次ls \n ...

  5. js模块系统 - amd|cmd|commonjs|esm|umd

    写过前端代码大概率听说过amd cmd umd commonjs esm这些名词, 想当初我第一次看到这些的时候, 人都麻了, 都是些啥啊. 后来我知道了, 这些都是js的模块规范. amd - 浏览 ...

  6. uniapp-scroll-view纵向(竖向)滑动当scrollTop为0时卡顿问题

    这个问题目前遇到的人少,所以找到答案不容易,我也是各种细节亲测才发现的解决方案.记录下来 当uniapp用scroll-view竖向滚动时,在scrollTop为0时,下拉会卡顿. 解决方法(只需要在 ...

  7. JavaCV的摄像头实战之七:推流(带声音)

    欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 本篇概览 本文是<JavaCV的摄像头实战> ...

  8. -2.输入加速(cin,cout)

    + ios::sync_with_stdio(false);//加速几百毫秒 cin.tie(0); // 接近scanf cout.tie(0);

  9. Gson解析:java.lang.IllegalArgumentException: declares multiple JSON fields named status 问题的解决

    在一次写定义系统统一返回值的情况下,碰到了java.lang.IllegalArgumentException: declares multiple JSON fields named status这 ...

  10. 变量命名 函数命名 方法 Naming cheatsheet

    Naming things is hard. This sheet attempts to make it easier. Although these suggestions can be appl ...