如何在循环内将大对象复制到omp任务中?
How to copy big object into omp task inside a loop?
我正在使用一个while循环来启动几个omp任务。每个任务都需要复制一个相当大的对象(作为第一个私有)。由于我的设置,大对象(在本例中为向量)将被天真地复制两次:
struct bigStruct {
bool next() {
/* do something with m_bigVector */
}
std::vector<int> m_bigVector;
/* other (big) data members */
};
bigStruct s;
#pragma omp parallel
{
#pragma omp single
while (s.next()) {
auto obj = s.m_bigVector; //copy the first time
#pragma omp task firstprivate(obj) //copy the second time
{
/* do something with obj */
}
}
} //end parallel
gcc优化(-O3)似乎并没有以任何方式优化这两个复制步骤。 一个(不太优雅的)解决方案是使用显式new/delete
:
#pragma omp parallel
{
#pragma omp single
while (s.next()) {
auto obj_ptr = new std::vector<int>(s.m_bigVector); //copy once
#pragma omp task firstprivate(obj_ptr) //copy only the pointer
{
/* do something with obj */
delete obj_ptr;
}
}
} //end parallel
有没有更modern/elegant
的方法来解决这一问题?也许有一种告诉任务移动对象而不是复制对象的方法? 请注意,我不想复制整个bigStruct
,因为它包含其他大数据成员。
好消息!
第一个私有变量不能具有引用类型。
自 OpenMP 4.5 (2015) 起已过时。目前没有这样的限制。有一个要求:
如果工作共享构造上的
firstprivate
子句中的列表项具有引用类型,则它必须绑定到团队所有线程的同一对象。
但这并不适用 -task
构造不是工作共享构造,无论如何都不会被多个线程遇到。
要充分了解标准的要求:
(关于列表项私有化)
如果列表项的类型是对类型
T
的引用,则该类型将被视为T
用于此子句的所有目的。将为构造分配具有自动存储持续时间的相同类型的新列表项。 这些列表项的存储和生存期一直持续到在其中创建它们的块退出为止。
对于类类型的每个变量:
• 如果
firstprivate
子句不在target
构造上,则调用复制构造函数来执行初始化
因此,您可以放心地执行以下操作:
auto& obj = s.m_bigVector;
#pragma omp task firstprivate(obj) // call copy ctor once
不幸的是,你不能
- 使用
const auto&
,因为这样obj
的类型将是 const,因为只删除引用。 - 将 obj 移动到
firstprivate
声明中。这很好,但仅适用于只有一个线程实际遇到数据共享子句的任务。
相关文章:
- C++17复制构造函数,在std::unordereded_map上进行深度复制
- 在C++程序中输入的文本文件将不起作用,除非文本被复制和粘贴
- 使用strcpy将char数组的元素复制到另一个数组
- 是否可以初始化不可复制类型的成员变量(或基类)
- 为什么在C++中使用私有复制构造函数与删除复制构造函数
- C++ Windows 驱动程序MSB3030无法复制该文件,因为它找不到
- 复制列表初始化的隐式转换的等级是多少
- 当从函数参数中的临时值调用复制构造函数时
- 有可能在Armadillo中复制MATLAB circshift方法吗
- 复制几乎为空的数组的最快方法
- 以下示例中如何避免代码复制?C++/库达
- 如果有一个模板构造函数只有一个泛型参数,为什么我必须有一个复制构造函数
- 为什么需要复制构造函数,在哪些情况下它们非常有用
- 不能将复制初始化与隐式转换的多个步骤一起使用
- 当有分配器意识的容器被复制/移动时,反弹分配器是否被复制/移走
- 为什么复制而不是移动数据元素?
- 文件系统:复制功能的速度秘诀是什么
- 使用仅使用一次的变量调用的复制构造函数.这可能是通过调用move构造函数进行编译器优化的情况吗
- 为什么类中的ostringstream类型的成员会导致";调用隐含删除复制构造函数";错误
- 如何在循环内将大对象复制到omp任务中?