我正在尝试实现Iterator模式。基本上,据我所知,它使类“可预见”,并使代码更安全,因为它不向用户透露确切的集合类型。
我已经做了一些实验,我发现如果我在我的类中实现IEnumerator GetEnumerator(),我会得到想要的结果.似乎省去了在实现接口方面的麻烦。
以下是我的意思的一瞥:
public class ListUserLoggedIn
{
/*
stuff
*/
public List<UserLoggedIn> UserList { get; set; }
public IEnumerator<UserLoggedIn> GetEnumerator()
{
foreach (UserLoggedIn user in this.UserList)
{
yield return user;
}
}
public void traverse()
{
foreach (var item in ListUserLoggedIn.Instance)
{
Console.Write(item.Id);
}
}
}我想我的问题是,这是Iterator的一个有效例子吗?如果是,为什么这是有效的,我如何使迭代器只返回一个部分或一个匿名对象通过"var“。如果没有,正确的方法是什么..。
发布于 2013-08-28 22:08:42
首先,一个更小、更简单的独立版本:
class Program
{
public IEnumerator<int> GetEnumerator() // IEnumerable<int> works too.
{
for (int i = 0; i < 5; i++)
yield return i;
}
static void Main(string[] args)
{
var p = new Program();
foreach (int x in p)
{
Console.WriteLine(x);
}
}
}奇怪的是,class Program没有实现IEnumerable。
Ecma-334的说明书上写着:
§8.18迭代器 foreach语句用于迭代可枚举集合的元素。为了可枚举,集合应具有返回枚举器的无参数GetEnumerator方法。
这就是为什么foreach()在你的课堂上工作的原因。没有提到IEnumerable。但是GetEnumerator()是如何产生实现Current和MoveNext的东西的呢?来自同一节:
迭代器是一个语句块,它产生一个有序的值序列。迭代器与普通语句块的区别在于存在一个或多个产生式语句。 重要的是要理解迭代器不是一种成员,而是实现函数成员的一种方法。
因此,方法的主体是迭代器块,编译器检查许多约束(该方法必须返回IEnumerable或IEnumerator),然后为您实现IEnumerator成员。
至于更深层次的“为什么”,我也学到了一些东西。根据Eric在“C#编程语言第三版”中的注释,第369页:
这被称为“基于模式的方法”,它可以追溯到之前的泛型。在C#1中,基于接口的方法完全基于传递
object,而值类型总是必须装箱的。模式方法允许foreach (int x in myIntCollection)没有仿制药也没有拳击。干净利落。
https://stackoverflow.com/questions/18498344
复制相似问题