std::condition_variable_any::wait_for
template<class Lock, class Rep, class Period > std::cv_status wait_for( Lock& lock, | (1) | (C++11以上) |
template<class Lock, class Rep, class Period, class Predicate > bool wait_for( Lock& lock, | (2) | (C++11以上) |
template<class Lock, class Rep, class Period, class Predicate > bool wait_for( Lock& lock, | (3) | (C++20以上) |
lock
を解放し、現在実行中のスレッドをブロックし、 *this の待機中スレッドのリストに追加します。 notify_all() または notify_one() が実行されるか、相対的なタイムアウト rel_time
が経過すると、スレッドのブロックは解除されます。 spurious にブロック解除される可能性もあります。 ブロックが解除されると、その理由にかかわらず、 lock
が再取得され、 wait_for()
を抜けます。 例外によってこの関数を抜ける場合も、 lock
は再取得されます。(C++14未満)標準は時間計測に steady clock を使用することを推奨しています。 スケジューリングやリソースの奪い合いによる遅延のため、この関数は timeout_duration
より長くブロックする可能性があります。
これらの関数が事後条件 (lock が呼び出し元スレッドによってロックされている) を満たせない場合は、 std::terminate が呼ばれます。 例えば、ミューテックスの再取得で例外が投げられると、これが発生し得ます。 | (C++14以上) |
目次 |
[編集]引数
lock | - | BasicLockable の要件を満たす Lock 型のオブジェクト。 現在のスレッドによってロックされていなければなりません |
stoken | - | 割り込みを登録する std::stop_token |
rel_time | - | 待機に費やす最大時間を表す std::chrono::duration 型のオブジェクト。 std::chrono::steady_clock::now() に加算したときにオーバーフローしない程度の小さな値でなければならないことに注意してください。 |
pred | - | 待機を続けるべき場合に false を返す述語。 述語関数のシグネチャは以下と同等であるべきです。 bool pred(); |
[編集]戻り値
pred
が rel_time
タイムアウト経過後も未だ false に評価される場合は false、そうでなければ true。[編集]例外
std::system_error を投げる場合があります。 lock.lock() または lock.unlock() によって投げられた例外を伝播する場合もあります。 | (C++14未満) |
実行中に clock、time_point、または duration によって投げられるあらゆる例外 (標準ライブラリによって提供される clock、time_point、および duration は、例外を投げることはありません)。 | (C++14以上) |
pred
によって投げられた例外を伝搬する場合もあります。[編集]ノート
ロック下で通知された場合であっても、タイムアウトで戻ったときは、オーバーロード (1) は紐付けられている述語の状態について、何の保証もしません。
notify_one()
/notify_all()
および wait()
/wait_for()
/wait_until()
の3つのアトミック部分 (unlock+wait、wakeup、lock) 各々の効果は、アトミック変数の変更順序として考えることができる単一の全順序で行われます。 この順序は個々の条件変数に固有です。 これにより、例えば notify_one()
が遅延され、 notify_one()
の呼び出しの後に待機を開始したスレッドをブロック解除することは、ありえなくなります。
[編集]例
#include <iostream>#include <atomic>#include <condition_variable>#include <thread>#include <chrono>usingnamespace std::chrono_literals; std::condition_variable_any cv;std::mutex cv_m;int i; void waits(int idx){std::unique_lock<std::mutex> lk(cv_m);if(cv.wait_for(lk, idx*100ms, []{return i ==1;}))std::cerr<<"Thread "<< idx <<" finished waiting. i == "<< i <<'\n';elsestd::cerr<<"Thread "<< idx <<" timed out. i == "<< i <<'\n';} void signals(){std::this_thread::sleep_for(120ms);std::cerr<<"Notifying...\n"; cv.notify_all();std::this_thread::sleep_for(100ms);{std::lock_guard<std::mutex> lk(cv_m); i =1;}std::cerr<<"Notifying again...\n"; cv.notify_all();} int main(){std::thread t1(waits, 1), t2(waits, 2), t3(waits, 3), t4(signals); t1.join(); t2.join(); t3.join(); t4.join();}
出力:
Thread 1 timed out. i == 0 Notifying... Thread 2 timed out. i == 0 Notifying again... Thread 3 finished waiting. i == 1
[編集]関連項目
条件変数が通知されるまで現在のスレッドをブロックします (パブリックメンバ関数) | |
条件変数が通知されるか指定時点に達するまで現在のスレッドをブロックします (パブリックメンバ関数) |