为什么D中的并行代码如此严重?

时间:2013-07-30 19:39:28

标签: c++ performance parallel-processing d

这是我在C ++和D中比较并行性时所做的一个实验。我使用相同的设计在两种语言中实现了一种算法(用于网络中社区检测的并行标签传播方案):并行迭代器获取句柄函数(通常一个闭包)并将其应用于图中的每个节点。

以下是D中的迭代器,使用taskPool中的std.parallelism实现:

/**
 * Iterate in parallel over all nodes of the graph and call handler (lambda closure).
 */
void parallelForNodes(F)(F handle) {
    foreach (node v; taskPool.parallel(std.range.iota(z))) {
        // call here
        handle(v);
    }
}

这是传递的句柄函数:

    auto propagateLabels = (node v){
        if (active[v] && (G.degree(v) > 0)) {
            integer[label] labelCounts;

            G.forNeighborsOf(v, (node w) {
                label lw = labels[w];
                labelCounts[lw] += 1; // add weight of edge {v, w}
            });

            // get dominant label
            label dominant;
            integer lcmax = 0;
            foreach (label l, integer lc; labelCounts) {
                if (lc > lcmax) {
                    dominant = l;
                    lcmax = lc;
                }
            }

        if (labels[v] != dominant) { // UPDATE
            labels[v] = dominant;
            nUpdated += 1; // TODO: atomic update?
            G.forNeighborsOf(v, (node u) {
                active[u] = 1;
            });
        } else {
            active[v] = 0;
        }

        }
    };

C ++ 11实现几乎相同,但使用OpenMP进行并行化。那么缩放实验表明了什么呢?

scaling

在这里,我检查弱缩放,将输入图形大小加倍,同时将线程数加倍并测量运行时间。理想情况是直线,但当然并行性有一些开销。我在main函数中使用defaultPoolThreads(nThreads)来设置D程序的线程数。 C ++的曲线看起来不错,但D的曲线看起来非常糟糕。我做错了什么吗? D并行性,或者这是否反映了并行D程序的可扩展性?

P.S。编译器标志

表示D:rdmd -release -O -inline -noboundscheck

for C ++:-std=c++11 -fopenmp -O3 -DNDEBUG

PPS。有些东西肯定是错的,因为D实现并行比顺序慢:

enter image description here

购买力平价。对于好奇,以下是两种实现的Mercurial克隆URL:

2 个答案:

答案 0 :(得分:8)

很难说因为我不完全理解你的算法应该如何工作,但看起来你的代码不是线程安全的,这导致算法运行的迭代次数超过了必要的。

我将此添加到PLP.run

的末尾
writeln(nIterations);

使用1个帖子nIterations = 19
有10个帖子nIterations = 34
使用100个帖子nIterations = 90

正如你所看到的那样,花费更长的时间不是因为std.parallelism的一些问题,而是因为它做了更多工作。

为什么您的代码不是线程安全的?

您并行运行的功能是propagateLabels,它具有对labelsnUpdatedactive共享,非同步访问权限。谁知道这造成了什么奇怪的行为,但它不是好事。

在开始分析之前,您需要将算法修复为线程安全的。

答案 1 :(得分:5)

正如彼得亚历山大指出的那样,你的算法似乎是线程不安全的。为了使其成为线程安全的,您需要消除可能同时或以未定义的顺序在不同线程中发生的事件之间的所有数据依赖性。一种方法是使用WorkerLocalStorage(在std.parallelism中提供)跨线程复制某个状态,并且可能在算法结束时以相对便宜的循环合并结果。

在某些情况下,复制此状态的过程可以通过将算法编写为缩减并使用std.parallelism.reduce(可能与std.algorithm.mapstd.parallelism.map组合)来自动执行升降。