在线程

时间:2017-07-30 02:04:20

标签: c++ c++11 stdthread

我需要在从基类方法中生成线程时调用纯虚方法实现,如下所示。

#include <iostream>

class Foo {
  private:
    std::thread tr;

  public:
    virtual void baz() = 0;
    void foo() {
      this->tr = std::thread([=] { this->baz(); });
    }

};

class Bar : public Foo {
  public:
    void baz() {
      std::cout << "In baz" << "\n";
    }

};

主要班级......

#include <thread>
#include "test.hpp"

int main(int argc, char* argv[]) {
  Bar b;
  b.foo();
}

但它失败并显示消息

  

在没有活动异常的情况下终止被调用

     

名为

的纯虚方法

消息“称为纯虚方法”仅出现在某些失败消息中。我究竟做错了什么?它是否与Bar相关或者线程被不正当地破坏了?

1 个答案:

答案 0 :(得分:2)

正如伊戈尔在评论中指出的那样,你有一场数据竞赛。该线程实际上是在Bar被销毁后执行的(当然,没有定义实际的执行顺序,所以有时你可能会很幸运)。为防止这种情况发生,您需要在tr.join();被销毁之前Bar

class Foo {
    std::thread tr;
protected:
    void join() { tr.join(); }
public:
    virtual ~Foo() = default;  // ~Foo should be virtual or protected, if Foo contains virtual methods
    virtual void baz() = 0;
    void foo() {
        this->tr = std::thread([=] { this->baz(); });
    }
};

class Bar : public Foo {
public:
    ~Bar() { join(); }
    void baz() { std::cout << "In baz" << "\n"; }
};

如果您想进一步调查此问题,请在各种方法中添加一些cout(尤其是析构函数),并在不同位置添加std::this_thread::sleep_for (std::chrono::seconds(1))