0

0

C++多线程如何使用互斥锁(mutex)?(代码示例)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-05 10:59:24

|

189人浏览过

|

来源于php中文网

原创

C++多线程中mutex用于防止数据竞争,核心是加锁→访问临界区→解锁;推荐用std::lock_guard或std::unique_lock自动管理;需避免死锁(统一加锁顺序)、条件等待用std::condition_variable配合while循环检查,且注意锁粒度与性能。

c++多线程如何使用互斥锁(mutex)?(代码示例)

在C++多线程中,mutex用于保护共享数据,防止多个线程同时访问导致数据竞争(data race)。核心是:**加锁 → 访问临界区 → 解锁**,推荐用 std::lock_guardstd::unique_lock 自动管理,避免忘记解锁或异常导致死锁。

基础用法:std::mutex + std::lock_guard

std::lock_guard 是最常用、最安全的方式——构造时自动加锁,析构时自动解锁(RAII机制),即使发生异常也不会遗漏解锁。

#include 
#include 
#include 
#include 

int counter = 0;
std::mutex mtx; // 全局互斥锁

void increment() {
    for (int i = 0; i < 100000; ++i) {
        std::lock_guard lock(mtx); // 自动加锁
        ++counter; // 临界区:仅此处访问共享变量
        // lock 析构时自动解锁(作用域结束即释放)
    }
}

int main() {
    std::vector threads;
    for (int i = 0; i < 4; ++i) {
        threads.emplace_back(increment);
    }
    for (auto& t : threads) t.join();
    std::cout << "Final counter: " << counter << "\n"; // 输出 400000
}

避免死锁:按固定顺序加锁

当一个线程需同时获取多个锁时,若不同线程以不同顺序请求锁,容易引发死锁。解决方法是统一加锁顺序,或使用 std::lock + std::adopt_lock

  • std::lock 同时锁定多个 mutex(内部使用死锁避免算法)
  • 再用 std::lock_guardstd::adopt_lock 标签接管已持锁
std::mutex mtx1, mtx2;
void transfer(int& from, int& to, int amount) {
    std::lock(mtx1, mtx2); // 安全地同时加锁
    std::lock_guard lock1(mtx1, std::defer_lock);
    std::lock_guard lock2(mtx2, std::defer_lock);
    // 更简洁写法(推荐):
    // std::lock_guard lock1(mtx1, std::adopt_lock);
    // std::lock_guard lock2(mtx2, std::adopt_lock);
    from -= amount;
    to += amount;
}

条件等待:配合 std::condition_variable

仅靠 mutex 无法实现“等待某条件成立再继续”,需搭配 std::condition_variable。注意:必须在持有对应 mutex 的前提下调用 wait

一览AI绘图
一览AI绘图

一览AI绘图是一览科技推出的AIGC作图工具,用AI灵感助力,轻松创作高品质图片

下载

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

  • wait 会自动释放锁并挂起线程;条件满足被唤醒后重新获取锁
  • notify_one()notify_all() 唤醒等待线程
  • 务必用 while 循环检查条件(防止虚假唤醒)
std::mutex mtx;
std::condition_variable cv;
bool ready = false;

void worker() {
    std::unique_lock lock(mtx);
    cv.wait(lock, []{ return ready; }); // 等待 ready == true
    std::cout << "Worker starts working...\n";
}

void notifier() {
    std::this_thread::sleep_for(std::chrono::seconds(1));
    {
        std::lock_guard lock(mtx);
        ready = true;
    }
    cv.notify_one(); // 唤醒一个等待线程
}

可重入与性能提示

std::mutex 不可重入(同一线程重复 lock 会死锁),如需可重入逻辑,应改用 std::recursive_mutex(但通常说明设计有问题)。

  • 粒度要合理:锁住的代码越少越好,避免长时间持有锁阻塞其他线程
  • 避免在临界区内做耗时操作(如 I/O、网络调用、复杂计算)
  • 考虑无锁替代方案(如 std::atomic)处理简单变量读写

不复杂但容易忽略。

相关专题

更多
while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

83

2023.09.25

线程和进程的区别
线程和进程的区别

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

475

2023.08.10

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

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

137

2025.12.24

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

393

2023.08.14

mc.js网页版入口地址大全
mc.js网页版入口地址大全

本专题整合了mc.js网页版入口地址大全以及mc.js1.8.8版本汇总,阅读专题下面的文章了解更多详细内容。

34

2026.01.05

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

18

2026.01.05

python处理大数据合集
python处理大数据合集

本专题整合了python处理大数据相关教程,阅读专题下面的文章了解更多详细内容。

49

2026.01.05

大数据专业学习教程
大数据专业学习教程

本专题整合了大数据专业学习相关教程,阅读专题下面的文章了解更多详细内容。

3

2026.01.05

python设置中文版教程合集
python设置中文版教程合集

本专题整合了python改成中文版相关教程,阅读专题下面的文章了解更多详细内容。

4

2026.01.05

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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