正确使用 C++ STL 线程的函子

correct use of functor for C++ STL thread

提问人: 提问时间:3/20/2017 更新时间:3/20/2017 访问量:5697

问:

我很难理解函数对象在 C++ STL 中作为线程例程的正确用法。根据我的理解,函子的好处之一是对象实例可以保持状态。有时我希望一个或多个线程运行一些例程并计算一些结果。然后,在加入线程后,我从对象中查询这些结果。我正在尝试对 C++ STL 线程做同样的事情并遇到一些问题。问题似乎源于 C++ STL 线程复制了我的对象,因此我不确定在加入线程时应该如何检查结果。下面是代码片段:

#include <iostream>
#include <thread>

using namespace std;

class Worker
{
public:
    Worker() : _value(0)
    {
    }

    void operator()(unsigned int value);

    unsigned int get_value() {return this->_value;}

private:
    unsigned int _value;
};

void Worker::operator()(unsigned int value)
{
    this->_value = value;
}

int main()
{
    Worker worker;
    thread thread(worker, 13);
    thread.join();
    unsigned int value = worker.get_value();
    cout << "value: " << value << endl;
}

上面的例子只是我遇到的问题的简单重现。我希望 worker.get_value() 返回 13,但它返回零。如何实例化具有状态的对象,让线程在该对象中运行例程,然后在线程完成后查询该对象的状态?

谢谢 缺口

C++ 多线程 STL 函子

评论


答:

4赞 Kerrek SB 3/20/2017 #1

不要复制,而是将线程绑定到引用:

thread thread(std::ref(worker), 13);
//            ^^^^^^^^
7赞 Slava 3/20/2017 #2

当您按值传递时,您将制作一个副本。因此,您可以通过引用包装器通过引用传递:

thread thread(std::ref(worker), 13);

或通过指针传递:

thread thread(&worker, 13);

在这两种情况下,您都必须确保对象生存期足够长。