最終更新日時(UTC):
が更新

履歴 編集

class
<shared_mutex>

std::shared_mutex(C++17)

namespace std {
  class shared_mutex;
}

概要

shared_mutexクラスは、Readers-writer lockパターンをサポートするミューテックスクラスである。

このパターンは、「複数のユーザーによる読み込みと、単一ユーザーによる書き込み」の排他制御を効率的に行う、というものである。

このミューテックスクラスのロック取得方法は2種類ある。

このクラスは、デストラクタで自動的にロックを手放すことはしない。そのため、以下の補助クラスを使用して、デストラクタで自動的にロックを手放す。

  • lock_guardクラス/unique_lockクラス:書き込み用のロックを自動的に手放す
  • shared_lock:読み込み用のロックを自動的に手放す

備考

  • このクラスは現状、書き込みロックと読み込みロックについてのスケジューリング戦略を規定せず、カスタマイズもできない。そのため、書き込みロックがなかなか取得できない、読み込みロックがなかなか取得できないというスタベーションの問題が発生した場合に、ユーザーの状況に合わせて戦略を変更することができない。POSIXのReaders-writer lockの実装では、スケジューリングのオプションを指定できる

メンバ関数

構築・破棄

名前 説明 対応バージョン
(constructor) コンストラクタ C++17
(destructor) デストラクタ C++17
operator=(const shared_mutex&) = delete; 代入演算子 C++17

排他の所有権

名前 説明 対応バージョン
lock 排他ロックを取得する C++17
try_lock 排他ロックの取得を試みる C++17
unlock 排他ロックを手放す C++17

共有の所有権

名前 説明 対応バージョン
lock_shared 共有ロックを取得する C++17
try_lock_shared 共有ロックの取得を試みる C++17
unlock_shared 共有ロックを手放す C++17

#include <iostream>
#include <thread>
#include <shared_mutex>
#include <chrono>

std::mutex print_mtx;
void print_value(int x)
{
  std::lock_guard<std::mutex> lock(print_mtx);
  std::cout << x << std::endl;
}

class X {
  std::shared_mutex mtx_;
  int count_ = 0;
public:
  // 書き込み側:カウンタを加算する
  void writer()
  {
    std::lock_guard<std::shared_mutex> lock(mtx_);
    ++count_;
  }

  // 読み込み側:カウンタの値を読む
  void reader()
  {
    int local_count;
    {
      std::shared_lock<std::shared_mutex> lock(mtx_);
      local_count = count_;
    } // 共有ロックをここで手放す
    print_value(local_count);
  }
};

X obj;
void writer_thread()
{
  for (int i = 0; i < 3; ++i) {
    std::this_thread::sleep_for(std::chrono::milliseconds(1));
    obj.writer();
  }
}

void reader_thread()
{
  for (int i = 0; i < 10; ++i) {
    std::this_thread::sleep_for(std::chrono::milliseconds(1));
    obj.reader();
  }
}

int main()
{
  // 書き込みユーザー1人
  // 読み込みユーザー3人
  std::thread writer1(writer_thread);
  std::thread reader1(reader_thread);
  std::thread reader2(reader_thread);
  std::thread reader3(reader_thread);

  writer1.join();
  reader1.join();
  reader2.join();
  reader3.join();
}

出力例

1
1
1
2
2
2
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3
3

バージョン

言語

  • C++17

処理系

関連項目

参照