在线程中调用纯虚方法
Invoking a pure virtual method within a thread
我需要在从基础 class 方法中生成线程时调用纯虚方法实现,如下所示。
#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";
}
};
主要class...
#include <thread>
#include "test.hpp"
int main(int argc, char* argv[]) {
Bar b;
b.foo();
}
但失败并显示消息
terminate called without an active exception
pure virtual method called
消息“调用纯虚方法”仅出现在部分失败消息中。我究竟做错了什么?是否与 Bar 或线程未正确销毁有关?
正如 Igor 在他的评论中指出的那样,您有一场数据竞赛。线程在 Bar
被销毁后才真正执行(当然,实际的执行顺序没有定义,所以有时你可能会很幸运)。为防止这种情况,您需要在 Bar
被销毁之前 tr.join();
。
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))
。
我需要在从基础 class 方法中生成线程时调用纯虚方法实现,如下所示。
#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";
}
};
主要class...
#include <thread>
#include "test.hpp"
int main(int argc, char* argv[]) {
Bar b;
b.foo();
}
但失败并显示消息
terminate called without an active exception
pure virtual method called
消息“调用纯虚方法”仅出现在部分失败消息中。我究竟做错了什么?是否与 Bar 或线程未正确销毁有关?
正如 Igor 在他的评论中指出的那样,您有一场数据竞赛。线程在 Bar
被销毁后才真正执行(当然,实际的执行顺序没有定义,所以有时你可能会很幸运)。为防止这种情况,您需要在 Bar
被销毁之前 tr.join();
。
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))
。