平时工作中我们经常用foreach来迭代一个集合。比如

 foreach (Student student in myClass)

Console.WriteLine(student);

基本所有的集合都能够foreach,但是必须要实现IEnumerable接口。IEnumerable接口很简单,就只有一个IEnumerator GetEnumerator() 方法。看这个方法的定义就知道,仅仅是公开了另一个接口IEnumerator。而IEnumerator才是真正的支持一个集合的迭代。IEnumerator有1个属性和2个方法。

public object Current;

public void Reset();

public bool MoveNext();

只允许读取集合的数据,而不允许修改。为了详细的讲解,我们来写一个简单的例子,就会一目了然。

首先我们创建一个学生类Student如下:

 using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace IenumerableDemo
{
public class Student
{
#region 私有变量 private readonly string _id;
private string _firstname;
private string _lastname; #endregion
#region 属性
public string ID { get { return _id; } } public string FirstName { get { return _firstname; } set { _firstname = value; } } public string LastName { get { return _lastname; } set { _lastname = value; } } #endregion #region 构造函数 public Student(string id, string firstname, string lastname)
{
this._id = id; this._firstname = firstname; this._lastname = lastname;
} #endregion
#region 重写基类object方法 public override string ToString()
{
return string.Format("{0} {1},ID:{2}", _firstname, _lastname, _id);
} public override bool Equals(object obj)
{
if (obj == null) return false;
if (Object.ReferenceEquals(this, obj)) return true;
if (this.GetType() != obj.GetType()) return false; Student objstudent = (Student)obj;
if (_id.Equals(objstudent._id)) return true; return false;
} public override int GetHashCode()
{
return _id.GetHashCode();
}
#endregion }
}

接下来我们定义一个ClassList类来承载学生。让我们先忘记Ienmerable。这个类包含一个ArrayList字段_student,在构造函数中模拟3个学生。_student是私有的,不对外公开的。

 using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace IenumerableDemo
{
public class ClassList
{ #region private Members private readonly string _id; private ArrayList _students; #endregion #region Properties
public string ID { get { return _id; } }
#endregion #region Constructors public ClassList(string id)
{ this._id = id;
_students = new ArrayList() { new Student("", "John", "Smith"), new Student("", "Jane", "Doe"), new Student("", "Bob", "Johnson") }; } #endregion }
}

为了让对象支持foreach 迭代,ClassList类需要实现IEnumerable。因为我们的student是存在ArrayList对象里的,而ArrayList类已经实现了IEnumerable,我们就可以使用ArrayList类的Ienumerable。

        public IEnumerator GetEnumerator()
{ return (_students as IEnumerable).GetEnumerator(); }

最终的代码贴一下:

 using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace IenumerableDemo
{
public class ClassList:IEnumerable
{ #region private Members private readonly string _id; private ArrayList _students; #endregion #region Properties
public string ID { get { return _id; } }
#endregion #region Constructors public ClassList(string id)
{ this._id = id;
_students = new ArrayList() { new Student("", "John", "Smith"), new Student("", "Jane", "Doe"), new Student("", "Bob", "Johnson") }; } #endregion public IEnumerator GetEnumerator()
{ return (_students as IEnumerable).GetEnumerator(); }
}
}

然后我们调用看看使用ArrayList的Ienumerable效果:

 using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace IenumerableDemo
{
class Program
{
static void Main(string[] args)
{
ClassList myClass = new ClassList("History 204"); foreach (Student student in myClass) Console.WriteLine(student); Console.ReadLine();
} } }

看来还是实现了效果。那么接下来我们看看自定义实现IEnumerable。实现IEnumerable其实只要实现IEnumerator接口就可以了。

我们创建我们自己的一个自定义类ClassEnumerator 来实现IEnumerator来完成和上面相同的结果。这个类基本上就只是通过_students的索引来进行迭代,Reset()方法就是把索引设置为-1.Current属性来获取当前的student,MoveNext()来跳到Current的下一个数据,并返回一个boolean来表示是否到了集合最后。

 using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace IenumerableDemo
{
public class ClassEnumerator : IEnumerator
{ private ClassList _classList; private int _index; public ClassEnumerator(ClassList classList)
{
this._classList = classList; _index = -;
} #region IEnumerator Members public void Reset()
{
this._index = -;
} public object Current
{
get { return _classList.Students[_index]; }
} public bool MoveNext()
{
_index++;
if (_index >= _classList.Students.Count)
return false;
else
return true; }
#endregion }
}

最后修改我们的ClassLst类:

 using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace IenumerableDemo
{
public class ClassList : IEnumerable
{ #region private Members private readonly string _id; private ArrayList _students; #endregion #region Properties
public string ID { get { return _id; } } public ArrayList Students { get { return _students; } }
#endregion #region Constructors public ClassList(string id)
{ this._id = id;
_students = new ArrayList() { new Student("", "John", "Smith"), new Student("", "Jane", "Doe"), new Student("", "Bob", "Johnson") }; } #endregion public IEnumerator GetEnumerator()
{ return (IEnumerator)new ClassEnumerator(this); }
}
}

可已看到还是相当简单的。运行结果和上面是一样的。

下来看看 foreach怎么工作。

其实foreach只是语法糖,最终会被CLR翻译成

             IEnumerator enumerator = myClass.GetEnumerator();
while (enumerator.MoveNext())
{
Console.WriteLine((Student)enumerator.Current); }

我们可以把foreach 换成这样试一下,结果是一样滴。

 using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace IenumerableDemo
{
class Program
{
static void Main(string[] args)
{
ClassList myClass = new ClassList("History 204"); //foreach (Student student in myClass) // Console.WriteLine(student); IEnumerator enumerator = myClass.GetEnumerator();
while (enumerator.MoveNext())
{
Console.WriteLine((Student)enumerator.Current); } Console.ReadLine();
} } }

在自己的对象里实现IEnumerator和IEnumerable的更多相关文章

  1. JavaScript怎么把对象里的数据整合进另外一个数组里

    https://blog.csdn.net/qq_26222859/article/details/70331833 var json1 = [ {"guoshui":[ 3000 ...

  2. vue 监听对象里的特定数据

    vue  监听对象里的特定数据变化 通常是这样写的,只能监听某一个特定数据 watch: { params: function(val) { console.log(val) this.$ajax.g ...

  3. vue :class 可以接收 字符串 数组 和 对象 对象里面的key值 根据true或false 显示不显示

    vue :class 可以接收 字符串 数组 和 对象 对象里面的key值 根据true或false 显示不显示 https://cn.vuejs.org/v2/guide/class-and-sty ...

  4. promise对象里resolve和reject状态讲解及Promise.all()的使用

    首先来说下同步异步与阻塞非阻塞的概念,同步异步与阻塞非阻塞并没有关系.同步异步主要是事情做完以后,如何进行处理.或者说关注的是一种消息通信机制. 同步的情况下,是由处理消息者自己去等待消息是否被触发: ...

  5. C# ~ 从 IEnumerable / IEnumerator 到 IEnumerable<T> / IEnumerator<T> 到 yield

    IEnumerable / IEnumerator 首先,IEnumerable / IEnumerator 接口定义如下: public interface IEnumerable /// 可枚举接 ...

  6. 2021年了,`IEnumerator`、`IEnumerable`还傻傻分不清楚?

    IEnumerator.IEnumerable这两个接口单词相近.含义相关,傻傻分不清楚. 入行多年,一直没有系统性梳理这对李逵李鬼. 最近本人在怼着why神的<其实吧,LRU也就那么回事> ...

  7. C#深度学习の枚举类型(IEnumerator,IEnumerable)

    一.关于枚举的含义 .Net提供了可枚举类型的接口IEnumerable和枚举器接口IEnumerator,程序集System.Collections 另: IQueryable 继承自IEnumer ...

  8. IEnumerator和IEnumerable详解

    IEnumerator和IEnumerable 从名字常来看,IEnumerator是枚举器的意思,IEnumerable是可枚举的意思. 了解了两个接口代表的含义后,接着看源码: IEnumerat ...

  9. 【安卓开发】为什么不能往Android的Application对象里存储数据

    在一个App里面总有一些数据需要在多个地方用到.这些数据可能是一个 session token,一次费时计算的结果等.通常为了避免activity之间传递对象的开销 ,这些数据一般都会保存到持久化存储 ...

随机推荐

  1. Openstack Murano(kilo)二次开发之添加Volume

    Openstack Murano(kilo)二次开发之添加Volume 欢迎转载,转载请注明出处:http://www.cnblogs.com/fmnisme/p/openstack_murano_a ...

  2. 关于解决keil4和mdk共存后51不能使用go to definition Of 'XXXXXX'问题

    自己安装keil4和mdk共存后,(我是先安装的keil 后安装的 MDK),在51单片机工程里不能使用go to definition Of 'XXXXXX'问题, 类似的如图 已经困扰了好长时间, ...

  3. CSS中一些不经意的细节问题1

    CSS这样的语法,细节问题非常多,往往一些难以处理的问题,有可能是一些细节问题不到位,所以先记下一些,留给以后自己看看. 1.line-height:150%与line-height:1.5 的区别 ...

  4. 【Android】混淆器(ProGuard)

    混淆器(ProGuard) 混淆器通过删除从未用过的代码和使用晦涩名字重命名类.字段和方法,对代码进行压缩,优化和混淆.结果是一个比较小的.apk文件,该文件比较难进行逆向工程.因此,当你的应用程序对 ...

  5. paip.python php的未来预测以及它们的比较优缺点

    paip.python php的未来预测以及它们的比较优缺点 跟个php比..python有下列的优点: 1.桌面gui 功能强大. 主要是pyqt很好...而ruby qt 则好像不更新了..php ...

  6. SQL Server 批量插入数据的方法

    运行下面的脚本,建立测试数据库和表. --Create DataBase create database BulkTestDB; go use BulkTestDB; go --Create Tabl ...

  7. jmap命令详解(转)

    1.命令基本概述 Jmap是一个可以输出所有内存中对象的工具,甚至可以将VM 中的heap,以二进制输出成文本.打印出某个java进程(使用pid)内存内的,所有‘对象’的情况(如:产生那些对象,及其 ...

  8. 关于python测试webservice接口的视频分享

    现在大公司非常流行用python做产品的测试框架,还有对于一些快速原型产品的开发也好,很好地支持OO编程,代码易读.Python的更新挺快的,尤其是第三方库. 对于测试人员,代码基础薄弱,用pytho ...

  9. 小白学数据分析----->什么才是留存率的关键?

    最近花了很多的时间在体验各种游戏,从火爆的卡牌,到策略,RPG等等,有一个问题在影响我,什么才是留存率的关键?今天就先讨论一些我的想法. 留存率已经成为大家最常提到的词汇,也是拿出来show一下的武器 ...

  10. android studio 乱码

    1. 设置 file- setting -file encodeing- 设置utf-8 2 .  build.gradle 添加 tasks.withType(JavaCompile) { opti ...