![](/img/trans.png)
[英]Is std::future::get() or std::future::wait() a replacement for std::thread::join()?
[英]std::thread: How to wait (join) for any of the given threads to complete?
例如,我有两个线程, t1
和t2
。 我想等t1
或 t2
完成。 这可能吗?
如果我有一系列线程,比如说std::vector<std::thread>
,我该怎么办呢?
总是使用std::condition_variable
等待并通知 , 例如 :
std::mutex m;
std::condition_variable cond;
std::atomic<std::thread::id> val;
auto task = [&] {
std::this_thread::sleep_for(1s); // Some work
val = std::this_thread::get_id();
cond.notify_all();
};
std::thread{task}.detach();
std::thread{task}.detach();
std::thread{task}.detach();
std::unique_lock<std::mutex> lock{m};
cond.wait(lock, [&] { return val != std::thread::id{}; });
std::cout << "Thread " << val << " finished first" << std::endl;
注意: val
不一定代表首先完成的线程,因为所有线程几乎同时完成并且发生覆盖migh,但它仅用于此示例的目的。
不,在C ++ 11的线程库中没有等待多个对象等效。
如果要等待一组操作中的第一个操作,请考虑让它们提供一个线程安全的生产者 - 消费者队列。
这是我发布的包含threaded_queue<T>
的帖子。 将线程的工作产品交付到这样的队列中。 让消费者阅读另一端。
现在有人可以一次等待多个线程的(工作产品)。 或一个线程。 或GPU着色器。 或者通过RESTful Web界面交付工作产品。 你不在乎。
线程本身应该由std::thread
类的线程池或其他更高级别的抽象来管理,因为std::thread
使得面向客户端的线程抽象不佳。
template<class T>
struct threaded_queue {
using lock = std::unique_lock<std::mutex>;
void push_back( T t ) {
{
lock l(m);
data.push_back(std::move(t));
}
cv.notify_one();
}
boost::optional<T> pop_front() {
lock l(m);
cv.wait(l, [this]{ return abort || !data.empty(); } );
if (abort) return {};
auto r = std::move(data.back());
data.pop_back();
return r;
}
void terminate() {
{
lock l(m);
abort = true;
data.clear();
}
cv.notify_all();
}
~threaded_queue()
{
terminate();
}
private:
std::mutex m;
std::deque<T> data;
std::condition_variable cv;
bool abort = false;
};
我在C ++ 17中使用std::optional
而不是boost::optional
。 它也可以用unique_ptr
或许多其他结构替换。
轮询等待很容易:
#include<iostream>
#include<thread>
#include<random>
#include<chrono>
#include<atomic>
void thread_task(std::atomic<bool> & boolean) {
std::default_random_engine engine{std::random_device{}()};
std::uniform_int_distribution<int64_t> dist{1000, 3000};
int64_t wait_time = dist(engine);
std::this_thread::sleep_for(std::chrono::milliseconds{wait_time});
std::string line = "Thread slept for " + std::to_string(wait_time) + "ms.\n";
std::cout << line;
boolean.store(true);
}
int main() {
std::vector<std::thread> threads;
std::atomic<bool> boolean{false};
for(int i = 0; i < 4; i++) {
threads.emplace_back([&]{thread_task(boolean);});
}
std::string line = "We reacted after a single thread finished!\n";
while(!boolean) std::this_thread::yield();
std::cout << line;
for(std::thread & thread : threads) {
thread.join();
}
return 0;
}
我在Ideone.com上获得的示例输出:
Thread slept for 1194ms.
We reacted after a single thread finished!
Thread slept for 1967ms.
Thread slept for 2390ms.
Thread slept for 2984ms.
这可能不是最好的代码,因为轮询循环不一定是最佳实践,但它应该作为一个开始。
没有标准的等待多线程的方法。
您需要使用Windows上的WaitForMultipleObjects
等特定于操作系统的功能。 仅限Windows的示例:
HANDLE handles[] = { t1.native_handle(), t2.native_handle(), };
auto res = WaitForMultipleObjects(2 , handles, FALSE, INFINITE);
有趣的是,当std::when_any
将被标准化时,人们可以做一个标准但浪费的解决方案:
std::vector<std::thread> waitingThreads;
std::vector<std::future<void>> futures;
for (auto& thread: threads){
std::promise<void> promise;
futures.emplace_back(promise.get_future());
waitingThreads.emplace_back([&thread, promise = std::move(promise)]{
thread.join();
promise.set_value();
});
}
auto oneFinished = std::when_any(futures.begin(), futures.end());
非常浪费,仍然没有,但标准。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.