- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我有一堆类可以 Process()
对象,并返回它们自己的对象:
public override IEnumerable<T> Process(IEnumerable<T> incoming) { ... }
我想编写一个处理器类来包装这些处理器之一,并记录包装的 Process()
方法可能抛出的任何未捕获的异常。我的第一个想法是这样的:
public override IEnumerable<T> Process(IEnumerable<T> incoming) {
try {
foreach (var x in this.processor.Process(incoming)) {
yield return x;
}
} catch (Exception e) {
WriteToLog(e);
throw;
}
}
但这行不通,因为 CS1626: Cannot yield a value in the body of a try block with a catch clause .
所以我想写一些概念上等价但可以编译的东西。 :-) 我有这个:
public override IEnumerable<T> Process(IEnumerable<T> incoming) {
IEnumerator<T> walker;
try {
walker = this.processor.Process(incoming).GetEnumerator();
} catch (Exception e) {
WriteToLog(e);
throw;
}
while (true) {
T value;
try {
if (!walker.MoveNext()) {
break;
}
value = walker.Current;
} catch (Exception e) {
WriteToLog(e);
throw;
}
yield return value;
}
}
但这比我希望的要复杂,我不能完全确定它的正确性或者没有更简单的方法。
我走在正确的轨道上吗?有没有更简单的方法?
最佳答案
如果您想做的是在处理枚举结果期间处理异常,那么您的尝试逻辑只需要直接进入您的 for/while 循环。
但是您的示例读起来好像您正在 try catch 并跳过枚举提供程序引发的异常。
据我所知,在 C# 中无法迭代枚举器并跳过枚举器本身内发生的异常。如果枚举器引发异常,则以后对 MoveNext() 的所有调用都将导致错误输出。
解释为什么会发生这种情况的最简单方法是使用这个非常简单的可枚举:
IEnumerable<int> TestCases()
{
yield return 1;
yield return 2;
throw new ApplicationException("fail eunmeration");
yield return 3;
yield return 4;
}
可以理解,当我们看这个例子时,很明显抛出的异常将导致整个 block 退出,并且永远不会处理第 3 和第 4 个 yield 语句。事实上,在第 3 个 yield 语句中会收到通常的“检测到无法访问的代码”编译器警告。
所以当可枚举更复杂时,同样的规则适用:
IEnumerable<int> TestCases2()
{
foreach (var item in Enumerable.Range(0,10))
{
switch(item)
{
case 2:
case 5:
throw new ApplicationException("This bit failed");
default:
yield return item;
break;
}
}
}
当引发异常时,此 block 的处理停止并将调用堆栈传递回最近的异常处理程序。
我在 SO 上找到的解决此问题的所有可行示例都没有继续枚举中的下一项,它们都在第一个异常处中断。
因此,要跳过枚举中的异常,您将需要提供程序来促进它。这只有在您对提供商进行编码的情况下才有可能,或者您可以联系这样做的开发人员,以下是如何实现此目的的过度简化示例:
IEnumerable<int> TestCases3(Action<int, Exception> exceptionHandler)
{
foreach (var item in Enumerable.Range(0, 10))
{
int value = default(int);
try
{
switch (item)
{
case 2:
case 5:
throw new ApplicationException("This bit failed");
default:
value = item;
break;
}
}
catch(Exception e)
{
if (exceptionHandler != null)
{
exceptionHandler(item, e);
continue;
}
else
throw;
}
yield return value;
}
}
...
foreach (var item in TestCases3(
(int item, Exception ex)
=>
Console.Out.WriteLine("Error on item: {0}, Exception: {1}", item, ex.Message)))
{
Console.Out.WriteLine(item);
}
这将产生以下输出:
0
1
Error on item: 2, Exception: This bit failed
3
4
Error on item: 5, Exception: This bit failed
6
7
8
9
我希望这能为 future 的其他开发人员解决这个问题,因为这是一个非常普遍的想法,一旦我们开始深入了解 Linq 和枚举,我们都会明白这一点。功能强大,但存在一些逻辑限制。
关于c# - 包装一个 IEnumerable 并捕获异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3835633/
任何人都可以向我解释 IEnumerable 和 IEnumerator 之间的区别是什么, 以及如何使用它们? 谢谢!!! 最佳答案 通常,一个 IEnumerable是可以枚举的对象,例如列表或数
function TSomething.Concat(const E: IEnumerable>): IEnumerable; begin Result := TConcatIterator.Cr
我正试图找到解决这个问题的办法: 给定一个 IEnumerable> 我需要一个返回输入的方法/算法,但是如果多个 IEnumerable 具有相同的元素,则每个巧合/组只返回一个。 例如 I
我有一个有趣的问题:给定一个 IEnumerable , 是否有可能产生 IEnumerable> 的序列一次将相同的相邻字符串分组? 让我解释一下。 1。基本说明示例: 考虑以下 IEnumerab
我有课 public class Test { public void M1(IEnumerable> p) { } public void M2(IEnumerable)> p) {
我尝试解决下一个练习: 输入:整数列表 count >= 1;一些正整数 k 输出:此整数的所有可能元组,长度为 k ; 例如 输入: {1, 2}; k = 4 输出: { {1, 1, 1, 1
抱歉奇怪的标题。我想要实现的目标很简单: IEnumerable> listoflist; IEnumerable combined = listoflist.CombineStuff(); 例子:
公共(public)类项目 { ... public class Order { public List Items ... } public class Customer {
我有一个 IEnumerable>我想转换为单一维度集合的集合。是否可以使用通用扩展方法来实现这一点?现在我正在这样做以实现它。 List filteredCombinations = new Lis
我有一个 IEnumerable> CustomObject在哪里有一个 x (用作键(在本例中为 1 、 2 、 3 ))和 y值(value)。一些假数据: { { {1, 2}, {2, 4
我需要做的是选择嵌套元素列表,这是我的查询 returns IEnumerable>这是我的 linq 表达式: from a in (questions.Select(x => x.AnswerLi
如何使用 LINQ(或其他方式)将 IEnumerables 的 IEnumerable 拆分为一个平面 IEnumerable? 最佳答案 enumerable.SelectMany(x => x)
例如: public interface IEnumerable { IEnumerator GetEnumerator(); } //This interface allows the c
我对 Reflection.Emit 有疑问。我想要动态创建的类,它具有 ICollection 的简单实现。我定义的所有方法都很好,而不是接下来的两个: public IEnumerator Get
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: Why was IEnumerable made covariant in C# 4? 我正在查看 MSDN
IEnumerator.MoveNext() 的实现是否预计会相对较快?或者如果“移动到下一项” 包括磁盘 IO、Web 请求或其他可能长时间运行的操作是否可以? 例如,我正在处理一个处理文档的项目,
以下代码创建了 List 的中间实例并在 yield 返回之前将值附加到它。有没有一种好的方法可以避免创建实例并直接 yield 返回单元格值? IEnumerable> GetStrValues()
我有两个 IEnumerable 对象,我想验证其中一个是否包含另一个的所有元素。 我正在使用 obj1.Intersect(obj2).Any() 但交集没有像我预期的那样工作。即使 obj2 中只
我正在尝试这个 MSDN page 上的例子.我试图更改 GetEnumerator 方法。我知道那似乎有些不对劲,但它符合要求然后就不会运行。错误是枚举器尚未启动,应该调用 MoveNext,但 它
我写过关于自定义 IEnumerator 的文章。从中生成 IEnumerable 的最简单方法是什么?理想的解决方案(一行代码)是是否有一些用于该目的的类。还是我必须自己创建? 最佳答案 不幸的是,
我是一名优秀的程序员,十分优秀!