- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有一个 for 循环,我想使其并行,但是线程必须共享一个 unordered_map
和一个 vector
。
因为 for 循环有点大,所以我将在这里发布一个简明的概述,以便我可以清楚地说明我的主要问题。请阅读评论。
unordered_map<string, vector<int>> sharedUM;
/*
here I call a function that updates the unordered_map with some
initial data, however the unordered_map will need to be updated by
the threads inside the for loop
*/
vector<int> sharedVector;
/*
the shared vector initially is empty, the threads will
fill it with integers, the order of these integers should be in ascending
order, however I can simply sort the array after all the
threads finish executing so I guess we can assume that the order
does not matter
*/
#pragma omp parallel for
for(int i=0; i<N; i++){
key = generate_a_key_value_according_to_an_algorithm();
std::unordered_map<string, vector<int>::iterator it = sharedUM.find(key);
/*
according to the data inside it->second(the value),
the thread makes some conclusions which then
uses in order to figure out whether
it should run a high complexity algorithm
or not.
*/
bool conclusion = make_conclusion();
if(conclusion == true){
results = run_expensive_algorithm();
/*
According to the results,
the thread updates some values of
the key that it previously searched for inside the unordered_map
this update may help other threads avoid running
the expensive algorithm
*/
}
sharedVector.push_back(i);
}
最初我保留了代码,所以我只是在 for 循环中使用了 #pragma
,但是我遇到了一些关于 sharedVector
更新的问题.所以我决定使用简单的锁来强制线程在写入 vector 之前获取锁。所以在我的实现中我有这样的东西:
omp_lock_t sharedVectorLock;
omp_init_lock(&sharedVectorLock);
...
for(...)
...
omp_set_lock(&sharedVectorLock);
sharedVector.push_back(i);
omp_unset_lock(&sharedVectorLock);
...
omp_destroy_lock(&sharedVectorLock);
我已经多次运行我的应用程序,一切似乎都运行良好,直到我决定自动重新运行它太多次,直到我得到错误的结果。因为我对 OpenMP 和一般线程的世界还很陌生,所以我不知道当写入者更新某些共享数据时我们应该锁定所有读取器这一事实。正如您在我的应用程序中看到的那样,线程总是从 unordered_map
中读取一些数据,以便得出一些结论并了解有关分配给它们的 key 的信息。但是,如果两个线程必须使用同一个键,并且当某个其他线程试图读取该键的值时,另一个线程已经达到更新这些值的地步,会发生什么情况?我相信这就是我的问题所在。
但是我现在的主要问题是我不确定避免此类事情发生的最佳方法是什么。这就像我的系统在 99% 的时间里都在工作,但是那 1% 的时间毁了一切,因为两个线程很少分配有相同的键,这反过来又是因为我的 unordered_map 通常很大。
锁定 unordered_map 会完成我的工作吗?最有可能,但效率不高,因为想要使用键 x
的线程 A
必须等待线程 B
已经在使用键 y
完成,其中 y
可以与 x
不同。
所以我的主要问题是,我应该如何解决这个问题?当且仅当两个线程使用相同的 key 时,我如何锁定 unordered_map
?
提前致谢
最佳答案
1 使用锁和互斥锁。您必须在并行 block 的外部声明并初始化锁变量(在#pragma omp parallel
之前),然后在并行 block 内部使用它们:(1) 获取锁 (如果另一个线程锁定了它,这可能会阻塞),(2)用竞争条件更改变量,(3)释放锁。最后,退出并行 block 后将其销毁。在并行 block 内声明的锁是线程本地的,因此不能提供同步。这可以解释您的问题。
2 写入复杂的 C++ 容器。 OpenMP 最初是为简单的 FORTRAN do 循环设计的(类似于 C/C++ for 具有整数控制变量的循环)。任何更复杂的事情都会让你头疼。为了安全起见,C++ 容器上的任何非常量操作都必须在锁内执行(对同一容器上的任何此类操作使用相同的锁)或 omp 临界区(使用同一容器上任何此类操作的相同名称)。这包括 pop()
和 push()
等,除了简单的读取。只有在此类非常量容器操作只占用一小部分时间的情况下,这才能保持高效。
3 如果我是你,我不会为 openMP 而烦恼(我用过它但现在很后悔)。对于 C++,您可以使用 TBB,它还带有一些线程安全但无锁的容器。它还允许您根据递归执行的任务而不是线程来思考(父任务产生子任务等),但是 TBB 有一些简单的并行 for 循环实现,例如。
关于c++ - OpenMP(C/C++) : Efficient way of sharing an unordered_map<string, vector<int>> 和线程之间的 vector<int>,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15855609/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!