- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我遇到了以下简单代码的段错误:
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"
#include <iostream>
#include <thread>
#include <unistd.h>
void run() {
sleep(1); // see below
cv::Mat source(10, 10, CV_32FC1, -1);
cv::Mat result(10, 10, CV_32FC1);
cv::Mat trX(result.rows, result.cols, CV_32FC1, 5);
cv::Mat trY(result.rows, result.cols, CV_32FC1, 5);
cv::remap(source, result, trX, trY, cv::INTER_LINEAR, cv::BORDER_TRANSPARENT);
std::cout << "done" << std::endl;
}
int main(int argc, char* argv[]) {
std::thread t1(run);
t1.join();
std::thread t2(run);
t2.join();
return 0;
}
如果我直接从 main()
中调用 run()
两次,根本不使用线程,它运行良好。如果我交换 t1.join();
和 std::thread t2(run);
(也就是说,在第一个线程完成之前启动第二个线程;这是 sleep
变得很重要),它也运行良好。
此外,如果我将 main
更改为
int main(int argc, char* argv[]) {
std::thread t1(run);
std::thread t2(run);
t1.join();
t2.join();
std::thread t3(run);
t3.join();
return 0;
}
它在第三个线程中出现段错误,但(奇怪的是)并非总是如此:大约有 2-3 次运行成功通过。但是,我无法成功运行带有上述两个线程的程序。
似乎source
、trX
和trY
中的特定值并不重要。
我在做的大程序在12月份运行正常,之后就没时间做了,但是更新了几次系统。现在大程序因完全相同的段错误而失败,所以我认为它应该与更新版本的 opencv 和/或 g++ 和/或 libstdc++ 相关。
是我的系统或我的代码有问题吗?或者这是一些已知的问题?或者我应该在哪里更好地报告?
我正在运行最新的 Ubuntu 16.10,g++ 6.2.0(我也尝试过 4.9,结果相同)。可能感兴趣的包的特定版本是:
$ dpkg-query -W -f='${binary:Package}\t${Version}\n' | grep -E '(g\+\+|c\+\+|opencv)'
g++ 4:6.1.1-1ubuntu2
g++-4.9 4.9.4-2ubuntu1
g++-5 5.4.1-2ubuntu2
g++-6 6.2.0-5ubuntu12
lib32stdc++6 6.2.0-5ubuntu12
libflac++6v5:amd64 1.3.1-4
libopencv-calib3d2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-contrib2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-core-dev:amd64 2.4.9.1+dfsg-2.1
libopencv-core2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-features2d2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-flann2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-highgui-dev:amd64 2.4.9.1+dfsg-2.1
libopencv-highgui2.4-deb0:amd64 2.4.9.1+dfsg-2.1
libopencv-imgproc-dev:amd64 2.4.9.1+dfsg-2.1
libopencv-imgproc2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-legacy2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-ml2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-objdetect2.4v5:amd64 2.4.9.1+dfsg-2.1
libopencv-video2.4v5:amd64 2.4.9.1+dfsg-2.1
libsigc++-2.0-0v5:amd64 2.8.0-2
libstdc++-4.9-dev:amd64 4.9.4-2ubuntu1
libstdc++-5-dev:amd64 5.4.1-2ubuntu2
libstdc++-6-dev:amd64 6.2.0-5ubuntu12
libstdc++6:amd64 6.2.0-5ubuntu12
libstdc++6:i386 6.2.0-5ubuntu12
我使用以下命令来构建代码:
g++ --std=c++14 test.cpp -lpthread -lopencv_highgui -lopencv_core -lopencv_imgproc -o test
Valgrind 输出:
==18499== Thread 2:
==18499== Invalid read of size 8
==18499== at 0x690F0BA: ??? (in /usr/lib/x86_64-linux-gnu/libtbb.so.2)
==18499== by 0x690F18A: ??? (in /usr/lib/x86_64-linux-gnu/libtbb.so.2)
==18499== by 0x6910CE7: ??? (in /usr/lib/x86_64-linux-gnu/libtbb.so.2)
==18499== by 0x690F691: ??? (in /usr/lib/x86_64-linux-gnu/libtbb.so.2)
==18499== by 0x690A01F: ??? (in /usr/lib/x86_64-linux-gnu/libtbb.so.2)
==18499== by 0x6908164: tbb::internal::allocate_root_with_context_proxy::allocate(unsigned long) const (in /usr/lib/x86_64-linux-gnu/libtbb.so.2)
==18499== by 0x51D9E21: cv::parallel_for_(cv::Range const&, cv::ParallelLoopBody const&, double) (in /usr/lib/x86_64-linux-gnu/libopencv_core.so.2.4.9)
==18499== by 0x55AE8A1: cv::remap(cv::_InputArray const&, cv::_OutputArray const&, cv::_InputArray const&, cv::_InputArray const&, int, int, cv::Scalar_<double> const&) (in /usr/lib/x86_64-linux-gnu/libopencv_imgproc.so.2.4.9)
==18499== by 0x1094AC: run() (in /home/petr/osm/draw/test/test)
==18499== by 0x10A360: void std::_Bind_simple<void (*())()>::_M_invoke<>(std::_Index_tuple<>) (in /home/petr/osm/draw/test/test)
==18499== by 0x10A2ED: std::_Bind_simple<void (*())()>::operator()() (in /home/petr/osm/draw/test/test)
==18499== by 0x10A2BD: std::thread::_State_impl<std::_Bind_simple<void (*())()> >::_M_run() (in /home/petr/osm/draw/test/test)
==18499== Address 0xfffffffffffffff7 is not stack'd, malloc'd or (recently) free'd
最佳答案
OpenCV 提供了一个 parallel_for_
允许使用计算机上可用的并行框架(Intel TBB、Pthreads 等)轻松并行部分代码的函数。
在您的情况下,您拥有的 OpenCV 版本似乎是 2.4.9
TBB 用作默认值 parallel_for_
后端。
Here从多个线程调用的 TBB 的另一个问题。解决方案可能是在从源代码构建 OpenCV 时禁用 TBB 并改用 Pthread(在 CMake 中禁用 TBB 并启用 Pthread)。
您的解决方案也应该没问题。用 setNumThreads(0)
,文档说:
If threads == 0, OpenCV will disable threading optimizations and run all it’s functions sequentially.
我猜 setNumThreads(1)
应该也可以吧?
很遗憾,我无法确定问题的确切根源:
cv::remap
总体上不是线程安全的还是仅使用 TBB?我做了两个测试:
OpenCV 3.2
来自 Ubuntu 16.04 上的源代码,Pthreads 用作 parallel_for_
后端OpenCV 3.2
来自 Ubuntu 16.04 上的源代码并改用 TBB在这两种情况下,我都没有遇到任何问题。所以我希望它已经在较新的 OpenCV 版本或较新的 TBB 版本中得到解决。
注意:你可以使用
std::cout << "getBuildInformation:\n" << cv::getBuildInformation() << std::endl;
打印 OpenCV 信息。在我的第二次测试中,我得到:
Parallel framework: TBB (ver 4.4 interface 9003)
关于c++ - cv::remap segfaults with std::thread,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42842661/
所以我需要一个简单的分配器来分配(有时使用清零)并随后从映射内存池中释放 4K block 。然而,在实现这个之后,在测试时我发现在释放一两个 block 之后,如果我尝试分配一个 block ,程序
我的任务是用 C 编写一个程序。该程序应该能够检查参数并创建与我提供的参数一样大的数组。我必须用随机数填充数组。到目前为止工作正常。稍后我的任务是使用指针对数组进行排序。第一件事是我不太明白指针是如何
我对 C 很陌生(仍然)所以如果我误解了一些基本的东西,请耐心等待 我有一个简单的程序,它应该将文件作为字符串读取,然后将该字符串拆分成行 - 将结果存储到 n 个字符串数组中。但是,当我运行以下代码
我在工作中使用的应用程序之一遇到了一个奇怪且烦人的问题。该应用程序是用 C++ 编写的,当应用程序终止(主函数返回或调用 exit)时,它会因段错误而崩溃。段错误似乎是由 basic_string 类
我使用 python swig 包装的 C++ 库。在它的 __init__.py 文件中,它 sets在导入包含实现代码的共享对象文件之前,使用 dlopen 标志 RTLD_GLOBAL。 这会导
我在这里遇到了段错误。我很困惑。请帮帮我。 f1 和 y 都是结构体节点的指针。我想把 y 的左转 f1 右转。 #include #include struct node{
我有一个在公共(public)结构中声明的数组,如下所示: uint16_t *registers; 在一个函数中,我正在检索一个字符字符串(存储在缓冲区中,请参阅下面的代码),其中包含以逗号分隔的数
我正在用 C 实现二叉搜索树。下面的代码工作正常,只是当我尝试从树中删除子树时得到 SEGFAULT: 源代码: #include #include struct node { int dat
struct vehicle *add_vehicle(struct vehicle *v){ struct vehicle *newcar = (struct vehicle*)malloc
我正在使用链接列表实现符号表,代码工作正常,但代码中存在内存泄漏, 我有以下结构 struct node { char* pcKey; void* pvValue; struct node
我正在尝试将字符串复制到数组并打印它。它适用于第一个 for 循环,但第二次出现 seg 错误。 main (int argc, char *argv[]){ int argcIndex; cha
自从我用 C 编写代码已经一年了,但我不明白为什么会出现段错误 // Assume all imports are made int printAgain(double** array, int si
这是我的代码。编辑:调用者包含在底部。 该函数读取数据文件,确定有多少行和列,然后将数据存储到 data_array 中。 int getdata(double* *data_array, int*
我认为有两组代码是等效的,但一组会导致段错误,而另一组则不会。我真的很困惑为什么会这样...... 我想创建一个查找函数 此代码确实有效: MyPair *> dummy(x, NULL);
希望有人能提供帮助。我可以毫无错误地编译,我没有发现任何语法错误,但是当我运行它时,它崩溃了。在启动时调试段错误。全面披露,这是作业。我不是要找人来编写这个代码,只是看看我的问题和我现有的代码,也许会
我正在尝试在OpenMP中并行化相当大的for-loop。大约有20%的时间运行正常,但其余时间会因各种段错误而崩溃,例如: *** glibc detected *** ./execute: dou
我有一个模板类 ISingleton class ISingleton { public: static T* getInstance() { lock_guard g
我正在为使用 LibSVM 的 Android 构建 NDK 应用程序。我在 XCode 中为我的 mac 构建了一个等价物(都是 C++) 我发现 Mac 可以高速准确地处理我给它的非常大的特征向量
我在 ARM linux 平台上有一个由简单代码引起的非常奇怪的崩溃。问题是它很少重现(一天一次),另一个问题是它在实际上无法重现的地方崩溃。 让我们从 C++ 代码开始。线程函数执行此操作:
我有这段代码 int main() { int *b = new int(8); cout<<" &b = "<
我是一名优秀的程序员,十分优秀!