質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
スレッドセーフ

マルチスレッド環境において、複数のスレッド上で常に正常に実行する事が可能なコードを、スレッドセーフなコードと呼びます。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

1回答

732閲覧

スレッドセーフになっていない

ddd8

総合スコア19

スレッドセーフ

マルチスレッド環境において、複数のスレッド上で常に正常に実行する事が可能なコードを、スレッドセーフなコードと呼びます。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2020/01/29 14:25

スレッドセーフなログクラスを作成しているのですが、スレッドセーフになっていない気がします。
以下のコードを実行したとき、テキストファイルに書かれるのは30行だと思うのですが、結果はそれより少ない行数になったりします。
どこが間違っているのかわからないでいます。アドバイスいただきたいです。

Log.h

c++

1#include <string> 2#include <mutex> 3 4class Log 5{ 6public: 7 static Log &GetInstance( 8 const std::wstring path, 9 const std::wstring filename 10 ); 11 12 void OutFile( 13 const std::wstring msg 14 ); 15 16private: 17 Log( 18 const std::wstring path, 19 const std::wstring filename 20 ); 21 ~Log() {}; 22 Log(const Log &other) {}; 23 Log& operator=(const Log &other) {}; 24 25private: 26 std::mutex m_mtx; 27 std::wstring m_file_location; 28 29};

Log.cpp

c++

1#include "Log.h" 2#include <fstream> 3 4Log &Log::GetInstance( 5 const std::wstring path, 6 const std::wstring name 7) { 8 static Log instance(path, name); 9 return instance; 10} 11 12Log::Log( 13 const std::wstring path, 14 const std::wstring name 15) { 16 m_file_location = path; 17 m_file_location += name; 18} 19 20void Log::OutFile( 21 const std::wstring msg 22) { 23 std::unique_lock<std::recursive_mutex>(m_mtx); 24 std::wofstream wfs; // wstring用filestreamの型 25 wfs.open(m_file_location, std::ios::app); 26 wfs << msg << std::endl; 27}

main.cpp

c++

1#include "log.h" 2#include <thread> 3 4 5int main() { 6 const std::wstring path = L".\"; 7 const std::wstring name = L"test.txt"; 8 9 static Log &log = Log::GetInstance(path, name); 10 11 std::thread t1([&](){ for (int i=0;i<10;i++) log.OutFile(L"Hello thread1!"); }); 12 std::thread t2([&](){ for (int i=0;i<10;i++) log.OutFile(L"Hello thread2!"); }); 13 std::thread t3([&](){ for (int i=0;i<10;i++) log.OutFile(L"Hello thread3!"); }); 14 15 t1.join(); 16 t2.join(); 17 t3.join(); 18 19 return 0; 20}

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

fukatani

2020/01/29 14:33

よさそうにも見えますが、、 出力も貼っていただけないでしょうか?
guest

回答1

0

ベストアンサー

C++

1std::unique_lock<std::recursive_mutex>(m_mtx);

これだとunique_lockの生存期間はこの式のみですから、生成して即座に破棄され、後続の行はスレッドセーフではありません。
ちゃんと名前を付けて変数にしてやれば、ブロックスコープになるので、ブロックを抜けるまでの間ロックされます。

C++

1std::unique_lock<std::recursive_mutex> lock(m_mtx); // 名前をつけてやる 2std::wofstream wfs; // wstring用filestreamの型 3wfs.open(m_file_location, std::ios::app); 4wfs << msg << std::endl;

投稿2020/01/29 15:14

編集2020/01/29 15:19
toki_td

総合スコア2850

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

ddd8

2020/01/29 20:35

なるほど… すみません。ありがとうございます!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問