4

计算for完成百分比很容易:

for (int i = 0; i < 10; i++)
{
    for (int j = 0; j < 10; j++)
    {
        // Do somethings...
        Console.WriteLine("{0}% Done!", i * 10 + j + 1);
    }
}

输出:

1% Done!
2% Done!
3% Done!
4% Done!
5% Done!
6% Done!
7% Done!
8% Done!
9% Done!
10% Done!
11% Done!
12% Done!
13% Done!
...
93% Done!
94% Done!
95% Done!
96% Done!
97% Done!
98% Done!
99% Done!
100% Done!

但我对计算完成百分比一无所知Parallel.For

Parallel.For(0, 10, i =>
{
    Parallel.For(0, 10, j =>
    {
        // Do somethings...
        Console.WriteLine("{0}% Done!", i * 10 + j + 1);
    });
});

输出:

1% Done!
2% Done!
3% Done!
51% Done!
52% Done!
53% Done!
54% Done!
55% Done!
56% Done!
57% Done!
58% Done!
59% Done!
60% Done!
4% Done!
5% Done!
6% Done!
...
69% Done!
70% Done!
71% Done!
72% Done!
73% Done!
74% Done!
75% Done!
76% Done!
77% Done!
78% Done!
79% Done!
80% Done!
36% Done!
37% Done!
38% Done!
39% Done!
40% Done!
35% Done!

输出未显示正确的完成百分比。

如何在Parallel.For不降低性能的情况下计算完成百分比?

4

2 回答 2

5

您可以为此使用Interlocked.Increment方法:

int progress = 0;
Parallel.For(0, 10, i =>
{
    Parallel.For(0, 10, j =>
    {
        // Do somethings...
        Console.WriteLine("{0}% Done!", Interlocked.Increment(ref progress));
    });
});

内部循环的每次迭代都会有一个增量。某些打印输出可能会出现乱序 - 具体来说,当线程在获得新的进度值之后但在打印完成之前被抢占时。

于 2014-03-11T19:04:21.707 回答
2

对于像这样的简单情况,当您知道您正在处理 100 个项目时,只需有一个您(线程安全地)递增并报告的外部变量:

int progress = 0;
Parallel.For(0, 10, i =>
{
    Parallel.For(0, 10, j =>
    {
        // Do somethings...
        Console.WriteLine("{0}% Done!", Interlocked.Increment(ref progress));
    });
});

但请注意,您可能仍然会得到一些乱序报告的值(因为并行任务将执行增量,然后必须竞争对 的访问Console,这是一个独占资源)

于 2014-03-11T19:04:44.137 回答