Попытка расширить в своих двух предыдущих вопросах Операции перемещения для класса с потоком в качестве переменной-члена и Вызов функции внутри лямбды, переданной в поток
Я не понимаю, почему поток, выполняющий wait_for, иногда не получает уведомления, что приводит к тупиковой ситуации. Cppreference говорит о переменных состояния http://en.cppreference.com/w/cpp/thread/condition_variable/notify_one
Уведомляющему потоку не нужно удерживать блокировку на том же мьютексе, что и у ожидающего (ых) потока (ов); на самом деле это пессимизация, поскольку уведомленный поток немедленно снова заблокируется, ожидая, пока уведомляющий поток снимет блокировку.
MCVE, прокомментированная строка объясняет, что меняется, если я удерживаю блокировку, но я не понимаю, почему:
#include <atomic>
#include <condition_variable>
#include <mutex>
#include <thread>
#include <iostream>
using namespace std;
class worker {
public:
template <class Fn, class... Args>
explicit worker(Fn func, Args... args) {
t = std::thread(
[&func, this](Args... cargs) -> void {
std::unique_lock<std::mutex> lock(mtx);
while (true) {
cond.wait(lock, [this]() -> bool { return ready; });
if (terminate) {
break;
}
func(cargs...);
ready = false;
}
},
std::move(args)...);
}
~worker() {
terminate = true;
if (t.joinable()) {
run_once();
t.join();
}
}
void run_once() {
// If i dont hold this mutex the thread is never notified of ready being
// true.
std::unique_lock<std::mutex> lock(mtx);
ready = true;
cout << "ready run once " << ready << endl;
cond.notify_all();
}
bool done() { return (!ready.load()); }
private:
std::thread t;
std::atomic<bool> terminate{false};
std::atomic<bool> ready{false};
std::mutex mtx;
std::condition_variable cond;
};
// main.cpp
void foo() {
worker t([]() -> void { cout << "Bark" << endl; });
t.run_once();
while (!t.done()) {
}
}
int main() {
while (true) {
foo();
}
return 0;
}