0

0

c++怎么使用mutex来保护共享数据_c++线程同步与互斥锁用法详解

尼克

尼克

发布时间:2025-11-10 13:45:03

|

304人浏览过

|

来源于php中文网

原创

使用std::mutex配合lock_guard或unique_lock可有效避免多线程数据竞争。1. std::mutex提供互斥访问,通过lock()/unlock()控制共享资源访问;2. 推荐使用std::lock_guard实现RAII管理,构造时加锁,析构时自动解锁,防止忘记释放;3. std::unique_lock更灵活,支持延迟加锁(如std::defer_lock)和手动控制锁范围;4. 示例中两个线程通过lock_guard保护counter变量,确保递增操作安全;5. unique_lock可用于需要分段加锁或与条件变量协作的场景;6. 注意避免死锁(统一加锁顺序)、减少临界区、禁止复制mutex。合理使用能显著提升线程安全性。

c++怎么使用mutex来保护共享数据_c++线程同步与互斥锁用法详解

在多线程编程中,多个线程同时访问共享数据容易导致数据竞争和不可预测的行为。C++ 提供了 std::mutex 来实现线程间的互斥访问,确保同一时间只有一个线程可以操作共享资源。下面详细介绍如何使用 mutex 保护共享数据。

什么是 std::mutex

std::mutex 是 C++11 引入的标准库类,定义在 头文件中,用于控制对共享资源的独占访问。调用它的 lock() 方法会阻塞其他试图加锁的线程,直到当前线程调用 unlock() 释放锁。

直接手动调用 lock 和 unlock 容易出错(比如忘记解锁),因此推荐配合 std::lock_guardstd::unique_lock 使用,它们利用 RAII(资源获取即初始化)机制自动管理锁的生命周期。

使用 std::lock_guard 保护共享数据

std::lock_guard 是最简单的 RAII 锁管理器。它在构造时加锁,析构时自动解锁,适合作用域明确的临界区。

立即学习C++免费学习笔记(深入)”;

示例:两个线程递增同一个计数器

MCP官网
MCP官网

Model Context Protocol(模型上下文协议)

下载
#include 
#include 
#include 

int counter = 0;
std::mutex mtx;

void increment() {
    for (int i = 0; i < 100000; ++i) {
        std::lock_guard lock(mtx); // 自动加锁
        ++counter; // 操作共享数据
    } // 超出作用域,自动解锁
}

int main() {
    std::thread t1(increment);
    std::thread t2(increment);

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

    std::cout << "Final counter value: " << counter << std::endl;
    return 0;
}

在这个例子中,每次进入 increment() 函数的循环体时,lock_guard 会尝试获取锁。由于 mutex 的存在,两个线程不会同时修改 counter,从而避免了数据竞争。

std::unique_lock 更灵活的锁管理

如果需要更精细的控制(如延迟加锁、条件变量配合等),可以使用 std::unique_lock。它支持移动语义,可以在运行时决定是否加锁,并允许手动调用 lock()unlock()

示例:使用 unique_lock 控制锁的范围

#include 
#include 
#include 

int shared_data = 0;
std::mutex mtx;

void update_data() {
    std::unique_lock ulock(mtx, std::defer_lock);
    // 此时并未加锁

    ulock.lock(); // 手动加锁
    shared_data += 100;
    std::cout << "Updated data to: " << shared_data << std::endl;
    ulock.unlock(); // 可以提前释放锁

    // 做一些不需要锁的操作
    std::this_thread::sleep_for(std::chrono::milliseconds(10));

} // ulock 析构,即使已 unlock 也不会重复释放

std::defer_lock 表示构造时不立即加锁,便于后续控制。这在需要分段加锁或与条件变量(std::condition_variable)配合时非常有用。

常见注意事项

  • 避免死锁:多个 mutex 时,始终按相同顺序加锁。例如线程 A 先锁 mtx1 再锁 mtx2,线程 B 也应如此,否则可能互相等待。
  • 不要在持有锁时执行耗时操作(如 I/O、网络请求),以免阻塞其他线程。
  • 尽量缩小临界区范围,只在真正访问共享数据时才加锁。
  • 禁止复制 mutex,std::mutex 是不可复制也不可移动的。

基本上就这些。合理使用 std::mutex 配合 std::lock_guardstd::unique_lock,能有效防止多线程环境下的数据竞争问题,提升程序稳定性。不复杂但容易忽略细节,写并发代码时要格外小心。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

472

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

107

2025.12.24

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

61

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

41

2025.12.31

视频文件格式
视频文件格式

本专题整合了视频文件格式相关内容,阅读专题下面的文章了解更多详细内容。

32

2025.12.31

不受国内限制的浏览器大全
不受国内限制的浏览器大全

想找真正自由、无限制的上网体验?本合集精选2025年最开放、隐私强、访问无阻的浏览器App,涵盖Tor、Brave、Via、X浏览器、Mullvad等高自由度工具。支持自定义搜索引擎、广告拦截、隐身模式及全球网站无障碍访问,部分更具备防追踪、去谷歌化、双内核切换等高级功能。无论日常浏览、隐私保护还是突破地域限制,总有一款适合你!

41

2025.12.31

出现404解决方法大全
出现404解决方法大全

本专题整合了404错误解决方法大全,阅读专题下面的文章了解更多详细内容。

198

2025.12.31

html5怎么播放视频
html5怎么播放视频

想让网页流畅播放视频?本合集详解HTML5视频播放核心方法!涵盖<video>标签基础用法、多格式兼容(MP4/WebM/OGV)、自定义播放控件、响应式适配及常见浏览器兼容问题解决方案。无需插件,纯前端实现高清视频嵌入,助你快速打造现代化网页视频体验。

9

2025.12.31

关闭win10系统自动更新教程大全
关闭win10系统自动更新教程大全

本专题整合了关闭win10系统自动更新教程大全,阅读专题下面的文章了解更多详细内容。

8

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 5.7万人学习

C 教程
C 教程

共75课时 | 3.8万人学习

C++教程
C++教程

共115课时 | 10.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号