C++多线程中互斥量std::mutex与模板类std::lock_guard

本文介绍C++中互斥量std::mutex的使用方法,包括如何通过成员函数lock()和unlock()实现资源锁定与解锁,以及如何利用std::lock_guard确保互斥量正确解锁,防止资源竞争。

一、 互斥量std::mutex
C++中通过实例化std::mutex创建互斥量实例,通过成员函数lock()对互斥量上锁,unlock()进行解锁。C++中与std::mutex相关的类(包括锁类型)和函数都声明在< mutex >头文件中,要使用std::mutex,就必须包含此头文件。

#include <mutex>
#include <thread>
#include <iostream>
 
int counter = 0;
std::mutex mtx; // 创建互斥量
 
void add()
{
    for (int i = 0; i < 1000000; i++)
    {
        mtx.lock(); // 加锁
        counter = counter + 1;
        mtx.unlock(); // 解锁
    }
}
 
void sub()
{
    for (int i = 0; i < 1000000; i++)
    {
        mtx.lock(); // 加锁
        counter = counter - 1;
        mtx.unlock(); // 解锁
    }
}
 
int main()
{
    std::thread t1(add);
    std::thread t2(sub);
 
    t1.join();
    t2.join();
 
    std::cout << "counter:\t" << counter << std::endl;
}

二、 模板类std::lock_guard
不过,实践中不推荐直接去调用成员函数,调用成员函数就意味着,必须在每个函数出口都要去调用unlock(),也包括异常的情况。C++标准库为互斥量提供了一个RAII语法的模板类std::lock_guard,在构造时就能提供已锁的互斥量,并在析构的时候进行解锁,从而保证了一个已锁互斥量能被正确解锁。下面的程序清单中,展示了如何在多线程应用中,使用std::mutex构造的std::lock_guard实例,对一个共享数据访问保护。std::mutex和std::lock_guard都在 < mutex >头文件中声明。

#include <mutex>
#include <thread>
#include <iostream>
 
int counter = 0;
std::mutex mtx; // 创建互斥量
 
void add()
{
    for (int i = 0; i < 1000000; i++)
    {
        std::lock_guard<std::mutex> lock(mtx); // 析构,自动解锁。
        counter = counter + 1;
    }
}
 
void sub()
{
    for (int i = 0; i < 1000000; i++)
    {
        std::lock_guard<std::mutex> lock(mtx); // 析构,自动解锁。
        counter = counter - 1;
    }
}
 
int main()
{
    std::thread t1(add);
    std::thread t2(sub);
 
    t1.join();
    t2.join();
 
    std::cout << "counter:\t" << counter << std::endl;
}

来源:http://www.xieyincai.com/20190401835.html

`std::lock_guard<std::mutex> lock(_mutex)` 是 C++11 引入的一种用于管理互斥锁mutex)的机制,它通过RAII(资源获取即初始化)的方式简化了对互斥锁的管理。 以下是对这一语句的解释: 1. **std::lock_guard**: 这是C++标准库中的一个模板类,用于在作用域内自动管理互斥锁。当一个 `std::lock_guard` 对象被创建时,它会尝试获取传入的互斥锁;当该对象被销毁时(例如,离开其作用域),它会自动释放互斥锁。 2. **std::mutex**: 这是C++标准库中用于线程同步的互斥锁类。互斥锁用于保护共享数据,防止多个线程同时访问导致的数据竞争和不一致。 3. **_mutex**: 这是一个 `std::mutex` 类型的变,表示需要被管理的互斥锁。 4. **lock(_mutex)**: 这里创建了一个 `std::lock_guard<std::mutex>` 对象,并将 `_mutex` 传递给它。这意味着 `_mutex` 被锁定,直到 `lock` 对象超出其作用域并被销毁为止。 使用 `std::lock_guard` 的好处在于,它可以确保即使在异常或提前返回的情况下,互斥锁也能被正确释放,从而避免死锁的发生。 示例代码如下: ```cpp #include <iostream> #include <thread> #include <mutex> std::mutex _mutex; void safe_print(const std::string& message) { // 创建一个 std::lock_guard 对象,自动管理互斥锁 std::lock_guard<std::mutex> lock(_mutex); // 此时 _mutex 已被锁定 std::cout << message << std::endl; // 当 lock 对象超出作用域时,_mutex 会被自动解锁 } int main() { std::thread t1(safe_print, "Hello from thread 1"); std::thread t2(safe_print, "Hello from thread 2"); t1.join(); t2.join(); return 0; } ``` 在这个例子中,两个线程会安全地打印消息,而不会因为并发访问而导致数据竞争。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值