在 C++ 11 线程中通过引用传递;变化是本地化的

passing by reference in C++ 11 threads; changes are localized

我正在尝试使用 C++ 11 线程来加快对我的 opencv 代码的一些处理。

同时我正在使用矢量来共享数据

请考虑以下一段代码:

start_t = cv::getTickCount();
std::vector <std::thread>   tMSE_Calc(nComparisons);
std::vector<float> sum;
sum.resize(nComparisons);

for(int i = 0 ; i < nComparisons ; ++i)
{
    tMSE_Calc[i] = std::thread(FindMSE_Fast, vector1, vector2, sum, i);
}

for(int i = 0 ; i < nComparisons ; ++i)
{
    tMSE_Calc[i].join();
}
end_t = cv::getTickCount();
logfile << "Actual Comparison Took " << (end_t - start_t) * freq * 1000 << " milliseconds"<< std::endl;

for(int i = 0 ; i < nComparisons ; ++i)
{
    std::cout << "Sum[" << i << "] = " << sum[i] << std::endl;
}

我有一个线程函数:

void FindMSE_Fast(cv::Mat& m1, cv::Mat& m2, std::vector<float>& sumVec, int idx)
{
    cv::Mat s;
    cv::absdiff(m1, m2, s);
    s = s.mul(s);
    cv::Scalar sum = cv::sum(s);
    sumVec[idx] = cv::sqrt(sum[0]);
    std::cout << "Difference is: " << cv::sqrt(sum[0]) << "index is: " << sumVec[idx] << std::endl;

    //std::cout << "Difference is: " << cv::sqrt(sum[0]) << std::endl;
}

在我看来很明显应该在线程调用结束时修改向量和,但是当我稍后在主线程中打印它时,它的所有内容都为零。

为了通过引用传递,我们需要对函数参数做一些不同的事情吗?

绑定的函数参数被复制,而不是通过引用传递。此行为与 std::bind() 相同。参数作为推导 T&& 传递给构造函数,但这仅用于初始化一些保存的值。

如果您不想复制参数,您可以传递一个 std::reference_wrapper<T>,例如,使用

std::thread(FindMSE_Fast, vector1, vector2, std::ref(sum), i)

同时通过引用传递 sum。由于 std::reference_wrapper<T> 有一个到 T& 的转换运算符,您的函数将被适当地调用。