在 foreach 循环中访问枚举器?
我有一个 List 类,我想重写 GetEnumerator()
以返回我自己的 Enumerator 类。该枚举器类将具有两个附加属性,这些属性将在使用枚举器时更新。
为了简单起见(这不是确切的业务案例),我们假设这些属性是 CurrentIndex
和 RunningTotal
。
我可以在 foreach 循环中手动管理这些属性,但我宁愿封装此功能以供重用,并且枚举器似乎是正确的位置。
问题: foreach 隐藏了所有的枚举器业务,那么有没有办法在 foreach 语句中访问当前的枚举器,以便我可以检索我的属性?或者我是否必须使用 foreach,使用令人讨厌的旧 while 循环,并自己操作枚举器?
I have a List class, and I would like to override GetEnumerator()
to return my own Enumerator class. This Enumerator class would have two additional properties that would be updated as the Enumerator is used.
For simplicity (this isn't the exact business case), let's say those properties were CurrentIndex
and RunningTotal
.
I could manage these properties within the foreach loop manually, but I would rather encapsulate this functionality for reuse, and the Enumerator seems to be the right spot.
The problem: foreach hides all the Enumerator business, so is there a way to, within a foreach statement, access the current Enumerator so I can retrieve my properties? Or would I have to foreach, use a nasty old while loop, and manipulate the Enumerator myself?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(3)
严格来说,我想说,如果您想完全按照您所说的进行操作,那么是的,您需要调用 GetEnumerator 并使用 while 循环自己控制枚举器。
在不太了解您的业务需求的情况下,您也许可以利用迭代器函数,例如这样的:
然后您可以执行以下操作:
更新:
好的,所以这是我所说的后续内容“鱼钩”解决方案。现在,让我添加一个免责声明,我实在想不出这样做的好理由,但您的情况可能有所不同。
这个想法是,您只需创建一个传递给迭代器函数的“鱼钩”对象(引用类型)。迭代器函数操作您的钓鱼钩对象,并且由于您仍然在外部代码中引用它,因此您可以了解正在发生的情况:
您可以像这样使用它:
本质上,FishingHook 对象使您可以控制如何迭代器执行。我从这个问题中得到的印象是,您需要某种方法来访问迭代器的内部工作原理,以便您可以在迭代过程中操纵它的迭代方式,但如果情况并非如此,那么这个解决方案可能对于你所需要的东西来说太过分了。
Strictly speaking, I would say that if you want to do exactly what you're saying, then yes, you would need to call GetEnumerator and control the enumerator yourself with a while loop.
Without knowing too much about your business requirement, you might be able to take advantage of an iterator function, such as something like this:
Then you can do this:
Updated:
Ok, so here's a follow-up with what I've termed my "fishing hook" solution. Now, let me add a disclaimer that I can't really think of a good reason to do something this way, but your situation may differ.
The idea is that you simply create a "fishing hook" object (reference type) that you pass to your iterator function. The iterator function manipulates your fishing hook object, and since you still have a reference to it in your code outside, you have visibility into what's going on:
You would utilize it like this:
Essentially, the FishingHook object gives you some control over how the iterator executes. The impression I got from the question was that you needed some way to access the inner workings of the iterator so that you could manipulate how it iterates while you are in the middle of iterating, but if this is not the case, then this solution might be overkill for what you need.
使用
foreach
,您确实无法获取枚举器 - 但是,您可以让枚举器返回(yield
)一个包含该数据的元组;事实上,您可能可以使用 LINQ 来为您完成此操作...(我无法使用 LINQ干净 获取索引 - 可以通过
Aggregate
,不过;所以这是元组方法)With
foreach
you indeed can't get the enumerator - you could, however, have the enumerator return (yield
) a tuple that includes that data; in fact, you could probably use LINQ to do it for you...(I couldn't cleanly get the index using LINQ - can get the total and current value via
Aggregate
, though; so here's the tuple approach)您还可以根据您的要求以更实用的方式执行类似的操作。您所要求的可以被认为是将多个序列“压缩”在一起,然后一次迭代它们。您给出的示例的三个序列是:
下一步是分别指定这些序列中的每一个:
最后一个更有趣......我能想到的两种方法是使用一个临时变量来对序列求和,或者重新计算每个项目的总和。第二个显然性能较差,我宁愿使用临时的:
最后一步是将这些全部压缩在一起。 .Net 4 将内置 内置 Zip 运算符,在这种情况下,它会看起来像这样:
当你尝试将越多的东西压缩在一起时,这显然会变得更加嘈杂。
在最后一个链接中,有您自己实现 Zip 功能的源代码。这确实是一小段简单的代码。
You can also do something like this in a more Functional way, depending on your requirements. What you are asking can be though of as "zipping" together multiple sequences, and then iterating through them all at once. The three sequences for the example you gave would be:
The next step would be to specify each of these sequences seperately:
The last one is more fun... the two methods I can think of are to either have a temporary variable used to sum up the sequence, or to recalculate the sum for each item. The second is obviously much less performant, I would rather use the temporary:
The last step would be to zip these all together. .Net 4 will have the Zip operator built in, in which case it will look like this:
This obviously gets noisier the more things you try to zip together.
In the last link, there is source for implementing the Zip function yourself. It really is a simple little bit of code.