スレッド間で共有されるリソースに対して複数のスレッドから同時に処理を行うと、プログラムの整合性が壊れてしまいます。これを防ぐために共有リソースに対する処理を適切に制御してプログラムの整合性を保つことを
"排他制御" といいます。
ミューテックスと呼ばれる種類のクラスによって共有リソースへアクセスできるスレッドを制限して、排他制御を実現できます。
std::mutex クラスは、スレッド間で排他的なロックの仕組みを提供するミューテックスです。
lock()
メンバ関数を呼び出すと、そのオブジェクトに対してロックをかけられます。既にほかのスレッドがロックをかけている場合は、そのロックが解除されるまで、呼び出し元スレッドの実行はブロックされます。
try_lock() メンバ関数を呼び出すと、ロックを試行できます。ロックの状態によって、以下のようになります。
unlock() メンバ関数を呼び出すと、オブジェクトに対してかけたロックを解除できます。
std::mutex クラスは、再帰的にロックをかけられません。1つのスレッド内で同じオブジェクトに対して lock()
メンバ関数を二度呼び出すと、デッドロックが発生します。
もしこのとき、処理系がデッドロックを検知できるならば、エラーコードに
std::errc::resource_deadlock_would_occur を設定した、
std::system_error
例外が送出されるかもしれません。
mutex と lock_guard を使って、シンプルな排他処理を行うプログラムを作成してみます。
自動的に unlock してくれるので、多くの場合でこの方法が最適です。
[環境]
| コンパイラ : | g++ (Ubuntu 13.3.0-6ubuntu2~24.04), | 13.3.0 |
| OS: | Ubuntu 24.04 (WSL), | |
最初に、うまく動かない例を示します。
こちら "bad_sample.cpp" はうまく動かない例です。4つのスレッドがそれぞれ 25,000回 のインクリメントをするので、最後に変数 count が 100,000 になることを期待するプログラムです。しかし同期処理がないために期待する動作をしない、という例です。
[プログラムソース] bad_sample.cpp
#include <iostream> // std::cout, std::endl
#include <thread> // std::thread
#include <cstdlib> // EXIT_SUCCESS
// グローバル変数
volatile long count = 0;
// カウントアップを行うスレッド関数
void CountThread(const size_t iterations)
{
int x;
for (size_t i = 0; i < iterations; ++i){
x = count;
// 問題を発生させにくい場合は次の行のコメントアウトを外してください。
// std::this_thread::yield(); // スレッドの処理明け渡し。c++11。
x++;
count = x;
}
}
// メイン関数
int main()
{
const size_t numThreads = 4;
std::thread countThreads[numThreads];
// スレッドを4つ作成
for (size_t i = 0; i < numThreads; ++i){
countThreads[i] = std::thread(CountThread, 25000);
}
// 4つのスレッドが終わるのを待つ
for (size_t i = 0; i < numThreads; ++i){
countThreads[i].join();
}
// 4つのスレッドが終了したのちの count の値をチェック
std::cout << "count = " << count << std::endl;
return EXIT_SUCCESS;
}
ビルドおよび実行結果:
$ g++ -std=c++17 -Wall -Wextra -Wpedantic -pthread -O2 bad_sample.cpp -o bad_sample.out $ ./bad_sample.out count = 72727 $
この実行例では、期待する 100,000 に対して実際には 72,727 という値になりました。
次に、mutex による同期処理を追加した例を紹介します。
[プログラムソース] good_sample_01.cpp
#include <iostream> // std::cout, std::endl #include <thread> // std::thread #include <cstdlib> // EXIT_SUCCESS #include <mutex> // mutex, lock_guard // グローバル変数 volatile long count = 0; // 共有変数 std::mutex count_mutex; // ミューテックス // カウントアップを行うスレッド関数 void CountThread(const size_t iterations) { int x; for (size_t i = 0; i < iterations; ++i){ // ミューテックスをロックして count にアクセスします。 std::lock_guard<std::mutex> guard(count_mutex); x = count; // 問題を発生させにくい場合は次の行のコメントアウトを外してください。 // std::this_thread::yield(); // スレッドの処理明け渡し。c++11。 x++; count = x; } } // メイン関数 int main() { const size_t numThreads = 4; std::thread countThreads[numThreads]; // スレッドを4つ作成 for (size_t i = 0; i < numThreads; ++i){ countThreads[i] = std::thread(CountThread, 25000); } // 4つのスレッドが終わるのを待つ for (size_t i = 0; i < numThreads; ++i){ countThreads[i].join(); } // 4つのスレッドが終了したのちの count の値をチェック std::cout << "count = " << count << std::endl; return EXIT_SUCCESS; }
ビルドおよび実行結果:
$ g++ -std=c++17 -Wall -Wextra -Wpedantic -pthread -O2 good_sample_01.cpp -o good_sample_01.out $ ./good_sample_01.out count = 100000 $
mutex による同期処理を加えることで、プログラム実行後の変数 count の値は期待する 100,000 となりました。
以上が mutex による同期処理を必要とする理由と効果の説明です。
| オプション | 説明 |
|---|---|
| -std=c++17 | C++17 標準でコンパイルします。言語機能(構文・ライブラリ仕様)が C++17 準拠になります。 |
| -Wall | 一般的な警告群を有効にします。コード品質の初期チェックに有用です。 |
| -Wextra | -Wall に含まれない追加の警告を有効にします。未使用パラメータなどを検出します。 |
| -Wpedantic | 規格に厳格に従っていない拡張や非標準的記法を警告します。移植性を高めたい場合に有効です。 |
| -pthread | POSIX threads のサポートを有効にし、スレッド関連のライブラリをリンクします。単に -lpthread をリンクするだけでなく、コンパイル時にスレッド対応(スレッドセーフな定義など)を有効にするフラグです。スレッド/ミューテックスなどを使うプログラムでは必須に近いオプションです(Linux)。 |
| -O2 | 最適化レベル 2 を指定します。実行速度向上のための最適化を行うが、"-O3" よりは穏やかで安定的な設定です。デバッグ時は "-g -O0" を推奨します。 |
std::mutex.lock, std::mutex.unlock を使って排他処理を実装する例を紹介します。
NOTE
lock 後の分岐や例外発生を漏れなく unlock しないとデッドロックを発生する方法です。特に理由なければ、基本的には前述の lock_guard による実装を推奨します。
[環境]
| コンパイラ : | g++ (Ubuntu 13.3.0-6ubuntu2~24.04), | 13.3.0 |
| OS: | Ubuntu 24.04 (WSL), | |
[プログラムソース] good_sample_02.cpp
#include <iostream> // std::cout, std::endl
#include <thread> // std::thread
#include <cstdlib> // EXIT_SUCCESS
#include <mutex> // mutex
// グローバル変数
volatile long count = 0; // 共有変数
std::mutex count_mutex; // ミューテックス
// カウントアップを行うスレッド関数
void CountThread(const size_t iterations)
{
int x;
for (size_t i = 0; i < iterations; ++i){
count_mutex.lock(); // ミューテックスをロックして count にアクセスします。
x = count;
// 問題を発生させにくい場合は次の行のコメントアウトを外してください。
// std::this_thread::yield(); // スレッドの処理明け渡し。c++11。
x++;
count = x;
count_mutex.unlock(); // ミューテックスのロックを解除します。
}
}
// メイン関数
int main()
{
const size_t numThreads = 4;
std::thread countThreads[numThreads];
// スレッドを4つ作成
for (size_t i = 0; i < numThreads; ++i){
countThreads[i] = std::thread(CountThread, 25000);
}
// 4つのスレッドが終わるのを待つ
for (size_t i = 0; i < numThreads; ++i){
countThreads[i].join();
}
// 4つのスレッドが終了したのちの count の値をチェック
std::cout << "count = " << count << std::endl;
return EXIT_SUCCESS;
}
ビルドおよび実行結果:
$ g++ -std=c++17 -Wall -Wextra -Wpedantic -pthread -O2 good_sample_02.cpp -o good_sample_02.out $ ./good_sample_02.out count = 100000 $
| オプション | 説明 |
|---|---|
| -std=c++17 | C++17 標準でコンパイルします。言語機能(構文・ライブラリ仕様)が C++17 準拠になります。 |
| -Wall | 一般的な警告群を有効にします。コード品質の初期チェックに有用です。 |
| -Wextra | -Wall に含まれない追加の警告を有効にします。未使用パラメータなどを検出します。 |
| -Wpedantic | 規格に厳格に従っていない拡張や非標準的記法を警告します。移植性を高めたい場合に有効です。 |
| -pthread | POSIX threads のサポートを有効にし、スレッド関連のライブラリをリンクします。単に -lpthread をリンクするだけでなく、コンパイル時にスレッド対応(スレッドセーフな定義など)を有効にするフラグです。スレッド/ミューテックスなどを使うプログラムでは必須に近いオプションです(Linux)。 |
| -O2 | 最適化レベル 2 を指定します。実行速度向上のための最適化を行うが、"-O3" よりは穏やかで安定的な設定です。デバッグ時は "-g -O0" を推奨します。 |
std::mutex::try_lock は、C++ の std::mutex クラスに属するメンバ関数で、ミューテックスをロックできるかどうかを試すための非ブロッキングな方法です。
使いどころ:
補足:
[プログラムソース] try-lock_01.cpp
#include <chrono>
#include <iostream> // std::cout
#include <mutex>
#include <thread>
std::chrono::milliseconds interval(100);
std::mutex mutex;
int job_shared = 0; // both threads can modify 'job_shared',
// mutex will protect this variable
int job_exclusive = 0; // only one thread can modify 'job_exclusive'
// no protection needed
// this thread can modify both 'job_shared' and 'job_exclusive'
void job_1()
{
bool bLoop = true;
std::this_thread::sleep_for(interval); // let 'job_2' take a lock
while (bLoop)
{
// try to lock mutex to modify 'job_shared'
if (mutex.try_lock())
{
std::cout << "job shared (" << job_shared << ")\n";
mutex.unlock();
bLoop = false; // finish after one successful modification;
}
else
{
// can't get lock to modify 'job_shared'
// but there is some other work to do
++job_exclusive;
std::cout << "job exclusive (" << job_exclusive << ")\n";
std::this_thread::sleep_for(interval);
}
}
return;
}
// this thread can modify only 'job_shared'
void job_2()
{
mutex.lock();
std::this_thread::sleep_for(5 * interval);
++job_shared;
mutex.unlock();
}
int main()
{
std::thread thread_1(job_1);
std::thread thread_2(job_2);
thread_1.join();
thread_2.join();
return 0;
}
ビルドおよび実行結果:
$ g++ -std=c++17 -Wall -Wextra -Wpedantic -pthread -O2 try-lock_01.cpp -o try-lock_01.out $ ./try-lock_01.out job exclusive (1) job exclusive (2) job exclusive (3) job exclusive (4) job shared (1) $
| オプション | 説明 |
|---|---|
| -std=c++17 | C++17 標準でコンパイルします。言語機能(構文・ライブラリ仕様)が C++17 準拠になります。 |
| -Wall | 一般的な警告群を有効にします。コード品質の初期チェックに有用です。 |
| -Wextra | -Wall に含まれない追加の警告を有効にします。未使用パラメータなどを検出します。 |
| -Wpedantic | 規格に厳格に従っていない拡張や非標準的記法を警告します。移植性を高めたい場合に有効です。 |
| -pthread | POSIX threads のサポートを有効にし、スレッド関連のライブラリをリンクします。単に -lpthread をリンクするだけでなく、コンパイル時にスレッド対応(スレッドセーフな定義など)を有効にするフラグです。スレッド/ミューテックスなどを使うプログラムでは必須に近いオプションです(Linux)。 |
| -O2 | 最適化レベル 2 を指定します。実行速度向上のための最適化を行うが、"-O3" よりは穏やかで安定的な設定です。デバッグ時は "-g -O0" を推奨します。 |
本ページの情報は、特記無い限り下記 MIT ライセンスで提供されます。
| MIT License Copyright (c) 2020 Kinoshita Hidetoshi Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |
| 2026-01-21 | - | "3._mutex::try_lock" を追加 |
| 2026-01-17 | - | "2._mutex.lock, mutex.unlock" を追加 |
| 2022-09-13 | - | ページデザイン更新 |
| 2020-06-13 | - | 新規作成 |