C++ 并发编程(二):Mutex(互斥锁)

C++ 并发编程(二):Mutex(互斥锁)多个线程访问同一资源时 为了保证数据的一致性 最简单的方式就是使用 mutex 互斥锁

大家好,欢迎来到IT知识分享网。

多个线程访问同一资源时,为了保证数据的一致性,最简单的方式就是使用 mutex(互斥锁)。

引用 cppreference 的介绍:

The mutex class is a synchronization primitive that can be used to protect shared data from being simultaneously accessed by multiple threads.

Mutex 1

直接操作 mutex,即直接调用 mutex 的 lock / unlock 函数。

#include <iostream> #include <mutex> #include <thread> #include <vector> std::mutex g_mutex; int g_count = 0; void Counter() { g_mutex.lock(); int i = ++g_count; std::cout << "count: " << i << std::endl; // 前面代码如有异常,unlock 就调不到了。 g_mutex.unlock(); } int main() { const std::size_t SIZE = 4; // 创建一组线程。 std::vector<std::thread> v; v.reserve(SIZE); for (std::size_t i = 0; i < SIZE; ++i) { v.emplace_back(&Counter); } // 等待所有线程结束。 for (std::thread& t : v) { t.join(); } return 0; }

可惜的是,STL 没有提供 boost::thread_group 这样代表一组线程的工具,通过 std::vector 固然也能达到目的,但是代码不够简洁。

Mutex 2

使用 lock_guard 自动加锁、解锁。原理是 RAII,和智能指针类似。

#include <iostream> #include <mutex> #include <thread> #include <vector> std::mutex g_mutex; int g_count = 0; void Counter() { // lock_guard 在构造函数里加锁,在析构函数里解锁。 std::lock_guard<std::mutex> lock(g_mutex); int i = ++g_count; std::cout << "count: " << i << std::endl; } int main() { const std::size_t SIZE = 4; std::vector<std::thread> v; v.reserve(SIZE); for (std::size_t i = 0; i < SIZE; ++i) { v.emplace_back(&Counter); } for (std::thread& t : v) { t.join(); } return 0; }

Mutex 3

使用 unique_lock 自动加锁、解锁。 unique_lock 与 lock_guard 原理相同,但是提供了更多功能(比如可以结合条件变量使用)。 注意:mutex::scoped_lock 其实就是 unique_lock<mutex> 的 typedef。

至于 unique_lock 和 lock_guard 详细比较,可移步 StackOverflow。

#include <iostream> #include <mutex> #include <thread> #include <vector> std::mutex g_mutex; int g_count = 0; void Counter() { std::unique_lock<std::mutex> lock(g_mutex); int i = ++g_count; std::cout << "count: " << i << std::endl; } int main() { const std::size_t SIZE = 4; std::vector<std::thread> v; v.reserve(SIZE); for (std::size_t i = 0; i < SIZE; ++i) { v.emplace_back(&Counter); } for (std::thread& t : v) { t.join(); } return 0; }

Mutex 4

为输出流使用单独的 mutex。 这么做是因为 IO 流并不是线程安全的! 如果不对 IO 进行同步,此例的输出很可能变成:

count == count == 2count == 41 count == 3 

因为在下面这条输出语句中:

std::cout << "count == " << i << std::endl;

输出 “count == ” 和 i 这两个动作不是原子性的(atomic),可能被其他线程打断。

#include <iostream> #include <mutex> #include <thread> #include <vector> std::mutex g_mutex; std::mutex g_io_mutex; int g_count = 0; void Counter() { int i; { std::unique_lock<std::mutex> lock(g_mutex); i = ++g_count; } { std::unique_lock<std::mutex> lock(g_io_mutex); std::cout << "count: " << i << std::endl; } } int main() { const std::size_t SIZE = 4; std::vector<std::thread> v; v.reserve(SIZE); for (std::size_t i = 0; i < SIZE; ++i) { v.emplace_back(&Counter); } for (std::thread& t : v) { t.join(); } return 0; }

免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://yundeesoft.com/165506.html

(0)
上一篇 2024-12-31 13:15
下一篇 2024-12-31 13:33

相关推荐

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注微信