Parallel.ForEach并且无法读取已关闭的TextReader异常

时间:2011-10-31 08:41:50

标签: c# task-parallel-library parallel.foreach

我有一个包含各种Feed网址的文本文件。我正在使用以下代码阅读集合中的所有URL(IEnumerable):

var URLs = File.ReadLines(Path.GetFullPath(@"Resources\FeedList.txt"));

在下一行中,我打印总计数:

Console.WriteLine("Total Number of feeds : {0}",URLs.Count());

之后我使用Parellel.ForEach构造,执行一些逻辑,对应每个URL。以下是我正在使用的代码:

Parallel.ForEach(URLs, (url) =>
                                       {
                                           // Some business logic
                                       });

问题是,一旦我添加代码来打印URL的数量,即调用的代码,URL对象上的Count()方法,我就会遇到异常。例外:

Total Number of feeds : 78

Unhandled Exception: System.AggregateException: One or more errors occurred. ---> System.ObjectDisposedException: Cannot read from a closed TextReader.
   at System.IO.__Error.ReaderClosed()
   at System.IO.StreamReader.ReadLine()
   at System.IO.File.<InternalReadLines>d__0.MoveNext()
   at System.Collections.Concurrent.Partitioner.DynamicPartitionerForIEnumerable`1.InternalPartitionEnumerator.GrabNextChunk(Int32 requestedChunkSize)
   at System.Collections.Concurrent.Partitioner.DynamicPartitionEnumerator_Abstract`2.MoveNext()
   at System.Threading.Tasks.Parallel.<>c__DisplayClass32`2.<PartitionerForEachWorker>b__30()
   at System.Threading.Tasks.Task.InnerInvoke()
   at System.Threading.Tasks.Task.InnerInvokeWithArg(Task childTask)
   at System.Threading.Tasks.Task.<>c__DisplayClass7.<ExecuteSelfReplicating>b__6(Object )
   --- End of inner exception stack trace ---
   at System.Threading.Tasks.Task.ThrowIfExceptional(Boolean includeTaskCanceledExceptions)
   at System.Threading.Tasks.Task.Wait(Int32 millisecondsTimeout, CancellationToken cancellationToken)
   at System.Threading.Tasks.Parallel.PartitionerForEachWorker[TSource,TLocal](Partitioner`1 source, ParallelOptions parallelOptions, Action`1 simpleBody, Action`2 bodyWi
   at System.Threading.Tasks.Parallel.ForEachWorker[TSource,TLocal](IEnumerable`1 source, ParallelOptions parallelOptions, Action`1 body, Action`2 bodyWithState, Action`3
   at System.Threading.Tasks.Parallel.ForEach[TSource](IEnumerable`1 source, Action`1 body)
   at DiscoveringGroups.Program.Main(String[] args) in C:\Users\Pawan Mishra\Documents\Visual Studio 2010\Projects\ProgrammingCollectiveIntelligence\DiscoveringGroups\Pro
Press any key to continue . . .

如果我删除/注释掉打印计数值的行,则Parallel.ForEach循环运行正常。

有没有人知道这里出了什么问题?

1 个答案:

答案 0 :(得分:4)

当您不需要时(或当类型明显多余时),请勿使用var。在这种情况下,它隐藏了正在发生的事情,你会对结果感到惊讶。

File.ReadLines方法不会读取所有行并返回一个集合,它会返回一个枚举器,当您从中获取项目时,它会读取这些行。它返回的类型不是字符串数组,而是IEnumerable<string>,如果您指定了变量的类型,您会注意到它:

string[] URLs = File.ReadLines(Path.GetFullPath(@"Resources\FeedList.txt"));

这会产生编译器错误,因为该方法不会返回数组,因此您可能会看到结果不符合您的预期。

当你在枚举器上使用Count()方法时,它将读取文件中的所有行以便对它们进行计数,所以当你稍后尝试再次使用枚举时,它已经读取了所有行并关闭了TextReader

使用File.ReadAllLines方法读取文件中的所有行,而不是获取枚举数:

string[] URLs = File.ReadAllLines(Path.GetFullPath(@"Resources\FeedList.txt"));

现在您可以多次使用该数组。