当前位置: 首页 > article >正文

C++中锁和互斥量的原理、区别和使用建议

image.png

在多线程编程中,锁和互斥量是两个非常重要的概念。它们都是用来解决线程同步问题的,但是它们的工作方式和使用场景有所不同。下面我们将详细介绍这两个概念。

互斥量(Mutex)

互斥量(Mutex)是一种用于保护共享资源的工具,它的名字来源于“互斥”(Mutual Exclusion)的概念。互斥量的工作原理是,任何时候只允许一个线程访问某个特定的资源。如果其他线程试图访问该资源,它们将被阻塞,直到拥有互斥量的线程释放资源。

在C++中,互斥量由std::mutex类表示,它提供了lock()unlock()两个方法来获取和释放互斥量。

std::mutex mtx;
mtx.lock();
// 访问共享资源
mtx.unlock();

锁(Lock)

锁是一种更高级的同步机制,它是建立在互斥量之上的。锁提供了一种自动管理互斥量的方式,使得在发生异常时能够自动释放互斥量,防止死锁。

在C++中,锁由std::lock_guardstd::unique_lock两个类表示。它们都需要在构造时传入一个互斥量,当锁对象的生命周期结束时,它会自动释放互斥量。

std::mutex mtx;
std::lock_guard<std::mutex> lock(mtx);
// 访问共享资源

在上面的代码中,即使在访问共享资源的过程中发生了异常,lock_guard对象在销毁时也会自动调用mtx.unlock(),确保互斥量被正确释放。

区别

  • 管理方式:互斥量需要手动管理,需要在正确的位置调用lock()unlock()方法。而锁则是自动管理,它会在构造时自动获取互斥量,在销毁时自动释放互斥量。

  • 异常安全:如果在互斥量保护的区域内发生异常,可能会导致互斥量没有被正确释放,从而引发死锁。而锁则可以保证在任何情况下都能正确释放互斥量。

  • 灵活性std::unique_lockstd::lock_guard更灵活,它允许延迟锁定、尝试锁定和可转移锁所有权。

示例

下面是一个使用互斥量和锁的例子,它演示了如何在多线程环境中保护共享资源。

#include <iostream>
#include <thread>
#include <mutex>

std::mutex mtx;
int shared = 0; // 共享资源

void increase_shared(int n) {
    std::lock_guard<std::mutex> lock(mtx);
    for (int i = 0; i < n; ++i) {
        ++shared;
    }
}

int main() {
    std::thread t1(increase_shared, 100000);
    std::thread t2(increase_shared, 100000);

    t1.join();
    t2.join();

    std::cout << "Shared value: " << shared << std::endl; // 输出:200000

    return 0;
}

在这个例子中,我们创建了两个线程,它们都试图增加共享资源的值。我们使用std::lock_guard来保护共享资源,确保任何时候只有一个线程能够访问它。

建议

在多线程编程中,我们通常更推荐使用锁,而不是直接使用互斥量,原因有以下几点:

  1. 异常安全:如果在互斥量保护的区域内发生异常,可能会导致互斥量没有被正确释放,从而引发死锁。而锁则可以保证在任何情况下都能正确释放互斥量。

  2. 自动管理:使用锁可以自动管理互斥量的生命周期,无需手动调用lock()unlock()方法,使代码更简洁,也更容易避免错误。

  3. 灵活性std::unique_lockstd::lock_guard更灵活,它允许延迟锁定、尝试锁定和可转移锁所有权。

然而,这并不是说我们完全不需要直接使用互斥量。在某些情况下,我们可能需要更细粒度的控制,这时候直接使用互斥量可能会更有用。例如,如果我们需要在多个操作之间保持锁定状态,或者需要在特定条件下释放锁定,那么直接使用互斥量可能会更方便。

总的来说,选择使用锁还是互斥量,主要取决于具体的需求和场景。在大多数情况下,使用锁可以提供更好的异常安全性和便利性。但在需要更细粒度控制的情况下,直接使用互斥量可能会更合适。例如,如果我们需要在多个操作之间保持锁定状态,或者需要在特定条件下释放锁定,那么直接使用互斥量可能会更方便。

以下是一个使用互斥量的示例,这个示例中,我们需要在两个操作之间保持锁定状态:

#include <iostream>
#include <thread>
#include <mutex>

std::mutex mtx;
int shared = 0; // 共享资源

void increase_shared(int n) {
    mtx.lock();
    for (int i = 0; i < n; ++i) {
        ++shared;
    }
    std::cout << "Shared value: " << shared << std::endl;
    mtx.unlock();
}

int main() {
    std::thread t1(increase_shared, 100000);
    std::thread t2(increase_shared, 100000);

    t1.join();
    t2.join();

    return 0;
}

在这个例子中,我们创建了两个线程,它们都试图增加共享资源的值。我们使用std::mutex来保护共享资源,确保任何时候只有一个线程能够访问它。在每个线程的操作完成后,我们打印出共享资源的值,然后释放互斥量。这种情况下,使用互斥量可以让我们更精确地控制锁定和解锁的时机。


http://www.kler.cn/a/502030.html

相关文章:

  • Android SDK下载安装(图文详解)
  • 【机器学习】主动学习-增加标签的操作方法-流式选择性采样(Stream-based selective sampling)
  • 整数和浮点数的存储
  • Docker 安装开源的IT资产管理系统Snipe-IT
  • 计算机网络 (40)域名系统DNS
  • MongoDB实践
  • 提升Docker运维效率:实用技巧与最佳实践
  • 【opencv】第8章 图像轮廓与图像分割修复
  • [Python学习日记-76] 网络编程中的 socket 开发 —— 介绍、工作流程、socket 模块用法和函数介绍
  • 云端 IPv4 VRRP+MSTP多备份组配置实验
  • oarcle执行报错提示:SQL 错误 [1839] [22008]: ORA-01839: 指定月份的日期无效问题解决
  • (免费送源码)计算机毕业设计原创定制:Java+ssm+MySQL 在线购票影城
  • 冲击全马330计划
  • Node.js 环境的管理服务工具
  • 一键获取Linux主机配置信息shell脚本,含网卡详情,网卡绑定等
  • 滑动窗口限流算法:基于Redis有序集合的实现与优化
  • Table-Augmented Generation(TAG):Text2SQL与RAG的升级与超越
  • springboot vue uniapp 仿小红书 1:1 还原 (含源码演示)
  • CVE-2025-22777 (CVSS 9.8):WordPress | GiveWP 插件的严重漏洞
  • 【机器学习】Kaggle实战Rossmann商店销售预测(项目背景、数据介绍/加载/合并、特征工程、构建模型、模型预测)
  • 无源器件-电容
  • Docker 安装开源的IT资产管理系统Snipe-IT
  • 高性能计算服务器是指什么?
  • 洛谷 P3853 [TJOI2007] 路标设置 C语言
  • 企业通过私有安全端点访问大型语言模型的益处
  • RNN之:LSTM 长短期记忆模型-结构-理论详解-及实战(Matlab向)