template <class Clock, class Duration>
bool try_lock_until(const chrono::time_point<Clock, Duration>& abs_time);
概要
タイムアウトする絶対時間を指定して排他ロックの取得を試みる
要件
この関数を呼び出したスレッドが、ミューテックスの排他所有権と共有所有権のいずれもを保持していないこと。
テンプレートパラメータ制約
chrono::is_clock_v<Clock>
がtrue
であること (C++20)
効果
abs_time
パラメータで指定された絶対時間に到達するまで、ミューテックスの排他所有権の取得を試みる。
排他所有権が取得できるまで、もしくはabs_time
時間に到達するまでこの関数はブロッキングする。
abs_time
にすでに到達していた場合、この関数はtry_lock()
と同じ効果をもち、ブロッキングせずにミューテックスの排他所有の権取得を試みる。
戻り値
排他所有権が取得できた場合はtrue
を返す。
abs_time
パラメータで指定された相対時間の間に排他所有権が取得できなかった場合はタイムアウトとなり、false
を返す。
例外
時計クラス、time_point
クラス、duration
クラスの構築が例外を送出する場合、この関数はそれらの例外を送出する。
備考
この関数の実装が、ミューテックスの排他所有権を保持しているスレッドがひとつもない場合でも、排他所有権の取得に失敗する可能性がある。
例
#include <thread>
#include <shared_mutex>
class X {
mutable std::shared_timed_mutex mtx_;
int value_ = 0;
public:
// メンバ変数value_への書き込みを排他的にする
void add_value(int value)
{
namespace chrono = std::chrono;
chrono::steady_clock::time_point tp = chrono::steady_clock::now();
// 排他ロックの取得を試みる(3秒後にタイムアウト)
if (!mtx_.try_lock_until(tp + std::chrono::seconds(3))) {
// 排他ロックの取得に失敗
std::error_code ec(static_cast<int>(std::errc::device_or_resource_busy), std::generic_category());
throw std::system_error(ec);
}
value_ = value;
mtx_.unlock(); // 排他ロックを手放す
}
// メンバ変数value_の値を読み込む
int get_value() const
{
int result = 0;
mtx_.lock_shared(); // 共有ロックを取得する
result = value_;
mtx_.unlock_shared(); // 共有ロックを手放す
return result;
}
};
int main()
{
X x;
std::thread t1([&x]{ x.add_value(1); int value = x.get_value(); });
std::thread t2([&x]{ x.add_value(2); int value = x.get_value(); });
t1.join();
t2.join();
}
出力
バージョン
言語
- C++14
処理系
- Clang: 3.5 ✅
- GCC: 4.9 ✅
- ICC: ??
- Visual C++: 2015 ✅