Zro*_*roq 2 c# parallel.foreach concurrent-queue
我有一个ConcurrentQueue,其中包含需要获取其来源的URL列表。当将Parallel.ForEach与ConcurrentQueue对象用作输入参数时,Pop方法将无效(应返回一个字符串)。
我使用MaxDegreeOfParallelism设置为四个的Parallel。我真的需要阻止并发线程的数量。使用具有并行性的队列是否多余?
提前致谢。
// On the main class
var items = await engine.FetchPageWithNumberItems(result);
// Enqueue List of items
itemQueue.EnqueueList(items);
var crawl = Task.Run(() => { engine.CrawlItems(itemQueue); });
// On the Engine class
public void CrawlItems(ItemQueue itemQueue)
{
Parallel.ForEach(
itemQueue,
new ParallelOptions {MaxDegreeOfParallelism = 4},
item =>
{
var worker = new Worker();
// Pop doesn't return anything
worker.Url = itemQueue.Pop();
/* Some work */
});
}
// Item Queue
class ItemQueue : ConcurrentQueue<string>
{
private ConcurrentQueue<string> queue = new ConcurrentQueue<string>();
public string Pop()
{
string value = String.Empty;
if(this.queue.Count == 0)
throw new Exception();
this.queue.TryDequeue(out value);
return value;
}
public void Push(string item)
{
this.queue.Enqueue(item);
}
public void EnqueueList(List<string> list)
{
list.ForEach(this.queue.Enqueue);
}
}
Run Code Online (Sandbox Code Playgroud)
如果您要做的ConcurrentQueue<T>一切就是不需要首先从单个线程向其中添加项目,然后在中进行迭代Parallel.ForEach()。正常List<T>就足够了。
另外,您对的实现ItemQueue非常可疑:
它继承ConcurrentQueue<string>并包含另一个ConcurrentQueue<string>。这没有多大意义,令人困惑且效率低下。
上的方法ConcurrentQueue<T>经过精心设计,以确保线程安全。您Pop()不是线程安全的。什么可能发生的是,你检查Count,发现它的1,然后调用TryDequeue(),并没有得到任何值(即value会null),因为另一个线程从两个电话之间的时间队列中删除的项目。
| 归档时间: |
|
| 查看次数: |
4530 次 |
| 最近记录: |