- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试通过使用 openMP 来加速我的 MPI 项目。我有一个包含 1000 个 2d 点的数据集,我正在使用强力算法来查找 2d 图中的最小距离。但是,当我尝试拆分执行线程时,它会极大地损害性能。如何正确使用 openMP?
这是我的尝试:
double calcDistance(double input[][2], int start, int stop){
double temp;
//declare and initialize minimum
double minimum = pow (((input[start+1][0]) - (input[start][0])),2) + pow(((input[start+1][1]) - (input[start][1])),2);
minimum = sqrt(minimum);
closestIndex1 = start;
closestIndex2 = start+1;
//Brute Force Algorithm to find minimum distance in dataset.
#pragma omp parallel for
for(int i=start; i<stop;i++){
for(int j=start; j<stop;j++){
temp = pow(((input[j][0]) - (input[i][0])),2) + pow(((input[j][1]) - (input[i][1])),2);
temp = sqrt(temp);
if(temp < minimum && i < j){
minimum = temp;
closestIndex1 = i;
closestIndex2 = j;
}//endif
}//end j
}//end i
return minimum;
}
我不得不说哇。谢谢,这非常有帮助,确实解决了我的一大堆问题。再次感谢你,gha.st。
最佳答案
首先,纯属运气您的程序看起来像这样工作。您确实存在数据竞争,这会导致我的机器上出现无效结果。考虑本文的以下测试工具:
::std::cout << ::xtd::target_info() << "\n\n"; // [target os] [target architecture] with [compiler]
static const int count = 30000;
auto gen = ::std::bind(::std::normal_distribution<double>(0, 1000), ::std::mt19937_64(42));
std::unique_ptr<double[][2]> input(new double[count][2]);
for(size_t i = 0; i < count; ++i)
{
input[i][0] = gen();
input[i][1] = gen();
}
::xtd::stopwatch sw; // does what its name suggests
sw.start();
double minimum = calcDistance(input.get(), 0, count);
sw.stop();
::std::cout << minimum << "\n";
::std::cout << sw << "\n";
在删除 omp pragma 的情况下执行函数时,其输出为:
Windows x64 with icc 14.0
0.0559233
7045 ms
或
Windows x64 with msvc VS 2013 (18.00.21005)
0.0559233
7272 ms
当完整地执行 omp pragma 时,它的输出是:
Windows x64 with icc 14.0
0.324085
675.9 ms
或
Windows x64 with msvc VS 2013 (18.00.21005)
0.0559233
4338 ms
由于机器使用 24 个线程(在启用 HT 的 12 个内核上),加速是显而易见的,但可能会更好,至少对于 msvc 而言。生成更快程序的编译器 (icc) 也通过给出每次运行都不同的错误结果来显示数据竞争。
注意:在为具有 10k 迭代的 x86 编译调试版本时,我还能够看到 msvc 的错误结果。
temp
代码中的变量的生命周期为最内层循环的一次迭代。通过移动其范围以匹配其生命周期,我们可以消除数据竞争的一个来源。我还冒昧地删除了两个未使用的变量并更改了 minimum
的初始化为常数:
double calcDistance(double input[][2], int start, int stop){
double minimum = ::std::numeric_limits<double>::infinity();
//#pragma omp parallel for // still broken
for(int i = start; i < stop; i++){
for(int j = start; j < stop; j++) {
double temp = pow(((input[j][0]) - (input[i][0])), 2) + pow(((input[j][2]) - (input[i][3])), 2);
temp = sqrt(temp);
if(temp < minimum && i < j) minimum = temp;
}
}
return minimum;
}
OMP 支持 reductions ,这很可能会表现得相当好。为了尝试它,我们将使用以下 pragma,它确保每个线程独立工作 minimum
使用最小运算符组合的变量:
#pragma omp parallel for reduction(min: minimum)
结果验证了 ICC 的方法:
Windows x64 with icc 14.0
0.0559233
622.1 ms
但是 MSVC 咆哮 error C3036: 'min' : invalid operator token in OpenMP 'reduction' clause
,因为它不支持最小缩减。为了定义我们自己的缩减,我们将使用一种称为 double-checked locking 的技术。 :
double calcDistance(double input[][2], int start, int stop){
double minimum = ::std::numeric_limits<double>::infinity();
#pragma omp parallel for
for(int i = start; i < stop; i++){
for(int j = start; j < stop; j++) {
double temp = pow(((input[j][0]) - (input[i][0])), 2) + pow(((input[j][1]) - (input[i][1])), 2);
temp = sqrt(temp);
if(temp < minimum && i < j)
{
#pragma omp critical
if(temp < minimum && i < j) minimum = temp;
}
}
}
return minimum;
}
这不仅是正确的,而且与 MSVC 的性能相当(请注意,这比不正确的版本快得多!):
Windows x64 with msvc VS 2013 (18.00.21005)
0.0559233
653.1 ms
ICC 的性能没有明显下降:
Windows x64 with icc 14.0
0.0559233
636.8 ms
虽然上面是串行代码的适当并行化,但考虑到您正在计算一大堆 temp
,它可以得到显着优化。由于您的 i < j
,您永远不会使用的结果条件。
通过简单地改变内循环的起点,不仅可以完全避免这种计算,还可以简化循环条件。
我们使用的另一个技巧是延迟 sqrt
计算到最后可能的一秒,因为它是一个同态变换,我们可以只对距离的平方进行排序。
最后,调用pow
因为正方形的效率相当低,因为它会产生大量我们不需要的开销。
这导致了最终的代码
double calcDistance(double input[][2], int start, int stop){
double minimum = ::std::numeric_limits<double>::infinity();
#pragma omp parallel for
for(int i = start; i < stop; i++) {
for(int j = i + 1; j < stop; j++) {
double dx = input[j][0] - input[i][0];
dx *= dx;
double dy = input[j][1] - input[i][1];
dy *= dy;
double temp = dx + dy;
if(temp < minimum)
{
#pragma omp critical
if(temp < minimum) minimum = temp;
}
}
}
return sqrt(minimum);
}
导致最终表现:
Windows x64 with icc 14.0
0.0559233
132.7 ms
或
Windows x64 with msvc VS 2013 (18.00.21005)
0.0559233
120.1 ms
关于c++ - 如何正确使用 openMP,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23439824/
这个问题已经有答案了: How to do case insensitive string comparison? (23 个回答) 已关闭 3 年前。 用户在我的输入栏中写入“足球”,然后执行第 6
啊,不习惯 javascript 中的字符串。 character_id= + id + correct= + correctOrIncorrect 这就是我需要制作成字符串的内容。如果您无法猜测字符
$(function() { var base_price = 0; CalculatePrice(); $(".math1").on('change', function(e) { Calc
我找不到任何文章回答问题:将Spinnaker部署到Spinnaker将管理的同一Kubernetes集群是否安全/正确?我主要是指生产,HA部署。 最佳答案 我认为Spinnaker和Kuberne
我正在使用MSVC在Windows上从源代码(官方源代码发布,而不是从仓库中)构建Qt5(Qt 5.15.0)。 我正在设置环境。变量,依赖项等,然后运行具有1600万个选项的configure,最后
我需要打印一个包含重复单词的数组。我的数组已经可以工作,但我不知道如何正确计算单词数。我已经知道,当我的索引计数器 (i) 为 49 时,并且当 (i) 想要计数到 50 时,我会收到错误,但我不知道
我正在遵循一个指南,该指南允许 Google map 屏幕根据屏幕尺寸禁用滚动。我唯一挣扎的部分是编写一个代码,当我手动调整屏幕大小时动态更改 True/False 值。 这是我按照说明操作的网站,但
我有一个类“FileButton”。它的目的是将文件链接到 JButton,FileButton 继承自 JButton。子类继承自此以使用链接到按钮的文件做有用的事情。 JingleCardButt
我的 friend 数组只返回一个数字而不是所有数字。 ($myfriends = 3) 应该是…… ($myfriends = 3 5 7 8 9 12). 如果我让它进入 while 循环……整个
这个问题在这里已经有了答案: Is there a workaround to make CSS classes with names that start with numbers valid?
我正在制作一个 JavaScript 函数,当调整窗口大小时,它会自动将 div 的大小调整为与窗口相同的宽度/高度。 该功能非常基本,但我注意到在调整窗口大小时出现明显的“绘制”滞后。在 JS fi
此问题的基本视觉效果可在 http://sevenx.de/demo/bootstrap-carousel/inc.carousel/tabbed-slider.html 获得。 - 如果你想看一看。
我明白,如果我想从函数返回一个字符串文字或一个数组,我应该将其声明为静态的,这样当被调用的函数被返回时,内容就不会“消亡”。 但我的问题是,当我在函数内部使用 malloc 分配内存时会怎样? 在下面
在 mySQL 数据库中存储 true/false/1/0 值最合适(读取数据消耗最少)的数据字段是什么? 我以前使用过一个字符长的 tinyint,但我不确定它是否是最佳解决方案? 谢谢! 最佳答案
我想一次读取并处理CSV文件第一行中的条目(例如打印)。我假设使用Unix风格的\n换行符,没有条目长度超过255个字符,并且(现在)在EOF之前有一个换行符。这意味着它是fgets()后跟strto
所以,我们都知道 -1 > 2u == true 的 C/C++ 有符号/无符号比较规则,并且我有一种情况,我想有效地实现“正确”比较。 我的问题是,考虑到人们熟悉的尽可能多的架构,哪种方法更有效。显
**摘要:**文章的标题看似自相矛盾。 本文分享自华为云社区《Java异常处理:如何写出“正确”但被编译器认为有语法错误的程序》,作者: Jerry Wang 。 文章的标题看似自相矛盾,然而我在“正
我有一个数据框,看起来像: dataDemo % mutate_each(funs(ifelse(. == '.', REF, as.character(.))), -POS) # POS REF
有人可以帮助我使用 VBScript 重新格式化/正确格式化带分隔符的文本文件吗? 我有一个文本文件 ^分界如下: AGREE^NAME^ADD1^ADD2^ADD3^ADD4^PCODE^BAL^A
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引起辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the he
我是一名优秀的程序员,十分优秀!