IEnumerable、foreach和GC

今天在某些代码中看到了一种对Dictionary的遍历方法

 private Dictionary<uint, uint> _SkillDicts = new Dictionary<uint, uint>();
 Dictionary<uint, uint>.Enumerator iter = _SkillDicts.GetEnumerator();
 while (iter.MoveNext()){
       //iter.Current.Key, iter.Current.Value的操作。
}

他用的是IEnumerable 与 IEnumerator的一种方法。于是去看了Dictionary的一些东西,于是发现了这个东西:

那他这样用就无可厚非了,但是我们不是有foreach吗?这样遍历对比于foreach它又有什么优缺点呢?于是乎又去看了下foreach:

在探讨foreach如何内部如何实现这个问题之前,我们需要理解两个C#里边的接口,IEnumerable 与 IEnumerator. 在C#里边的遍历集合时用到的相关类中,IEnumerable是最基本的接口。这是一个可以进行泛型化的接口,比如说IEnumerable<User>.在微软的.NET推出了这两个接口后,才有了foreach的用法,可以说,foreach是建立在这两个接口的基础之上的,foreach的前提是其里边的容器要实现了IEnumerable接口。

IEnumerable 这个接口里边定义的内容非常简单,最重要的就是里边有一个抽象方法GetEnumerator. IEnumerable的意思是这个集合是可以遍历的,而这个GetEnumerator方法返回的IEnumerator的就是一个遍历器,用这个工具来遍历这个类。如果说IEnumerable 是一瓶香槟,那么IEnumerator就是一个开瓶器。在实现这个IEnumerable接口的时候,必须要实现这个GetEnumerator方法,要返回一个实例化的IEnumorator. 

foreach (Person p in peopleList)  
   Console.WriteLine(p.firstName + " " + p.lastName);  
//翻译成  
  
IEnumerator enumerator = (peopleList).GetEnumerator();  
try {  
      while (enumerator.MoveNext()) {  
      Person element; //post C# 5  
      element = (Person )enumerator.Current;  
    //下边这句就是原来foreach方法体中的逻辑  
      Console.WriteLine(p.firstName + " " + p.lastName);  
   }  
}  
finally {  
   IDisposable disposable = enumerator as System.IDisposable;  
   if (disposable != null) disposable.Dispose();  
} 

大概是因为多了个IDisposable的东西。。

对于垃圾回收而言,在C#中,托管资源的垃圾回收是通过CLR的Garbage Collection来实现的,Garbage Collection会调用堆栈上对象的析构函数完成对象的释放工作;而对于一些非托管资源,比如数据库链接对象等,需要实现IDisposable接口进行手动的垃圾回收。。 在C#1中已经内建了对迭代器的支持,那就是foreach语句。使得能够进行比for循环语句更直接和简单的对集合的迭代,编译器会将foreach编译来调用GetEnumerator和MoveNext方法以及Current属性,如果对象实现了IDisposable接口,在迭代完成之后会释放迭代器。

原来我们平常事的foreach产生GC的原因大概就是这个东西了。。

感觉还不够深入,后续再补充

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`IEnumerator` 和 `IEnumerable` 两个接口是 C# 中用于实现迭代器的关键接口。 `IEnumerable` 接口有一个方法 `GetEnumerator()`,它返回一个实现了 `IEnumerator` 接口的对象,该对象可以迭代集合中的元素。实现了 `IEnumerable` 接口的类可以使用 `foreach` 循环迭代集合中的元素。 `IEnumerator` 接口定义了三个方法: - `MoveNext()`:将枚举数推进到集合的下一个元素。 - `Reset()`:将枚举数重置为其初始位置,即在集合中第一个元素之前。 - `Current`:获取集合中的当前元素。 `IEnumerator` 接口的实现类需要在 `MoveNext()` 方法中实现迭代器的逻辑,并在 `Current` 属性中返回当前元素的值。 示例代码: ```csharp public class MyCollection : IEnumerable { private int[] array = { 1, 2, 3, 4 }; public IEnumerator GetEnumerator() { return new MyEnumerator(array); } private class MyEnumerator : IEnumerator { private int[] array; private int position = -1; public MyEnumerator(int[] array) { this.array = array; } public bool MoveNext() { position++; return (position < array.Length); } public void Reset() { position = -1; } public object Current { get { try { return array[position]; } catch (IndexOutOfRangeException) { throw new InvalidOperationException(); } } } } } ``` 上述代码中,`MyCollection` 类实现了 `IEnumerable` 接口,并在 `GetEnumerator()` 方法中返回了一个实现了 `IEnumerator` 接口的对象。`MyEnumerator` 类实现了 `IEnumerator` 接口,并在 `MoveNext()` 方法中实现了迭代器的逻辑,同时在 `Current` 属性中返回当前元素的值。这样,我们就可以通过 `foreach` 循环来迭代 `MyCollection` 类中的元素了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值