C++ 多线程与并发系统取向(二)—— 资源保护:std::mutex 与 RAII(类比 Java synchronized)

一、先问一个工程问题

你有一个共享容器:

cpp 复制代码
std::vector<int> data;

两个线程同时往里面 push_back()

会发生什么?

你可能会说:

vector 不是自动扩容的吗?

是的,但:

vector 不是线程安全的。

内部会:

  • 修改 size
  • 可能 realloc
  • 移动内存

如果两个线程同时写:

未定义行为。

二、资源保护的本质

并发系统里最重要一句话:

不是"给代码加锁",而是"给资源加保护策略"。

资源 = 被多个线程共享的数据。

三、什么是临界区(Critical Section)

临界区是:

访问共享资源的代码区域。

例如:

cpp 复制代码
data.push_back(1);

这一行,就是临界区。

四、最原始写法(危险)

cpp 复制代码
std::mutex mtx;

void task() {
    mtx.lock();
    data.push_back(1);
    mtx.unlock();
}

看起来没问题。

但如果中间抛异常?

cpp 复制代码
mtx.lock();
throw std::runtime_error("error");
mtx.unlock();  // 永远执行不到

锁永远不释放。

其他线程:

永久阻塞(死锁)。

五、C++ 的核心思想:RAII

RAII = Resource Acquisition Is Initialization

意思:

资源的获取与对象生命周期绑定。

锁对象在构造时加锁,在析构时自动解锁。

这就是:

cpp 复制代码
std::lock_guard

六、正确写法:std::lock_guard

cpp 复制代码
std::mutex mtx;

void task() {
    std::lock_guard<std::mutex> lock(mtx);
    data.push_back(1);
}

作用:

  • 构造 → 自动 lock
  • 离开作用域 → 自动 unlock

无论:

  • return
  • 异常
  • break

都会释放锁。

七、Java 对比:synchronized

Java 写法:

cpp 复制代码
synchronized (lock) {
    data.add(1);
}

本质类似:

  • 进入代码块 → 加锁
  • 离开代码块 → 自动释放

区别:

C++ Java
lock_guard 是对象 synchronized 是语言关键字
手动控制 mutex JVM 管理 monitor

八、完整并发示例(安全版)

cpp 复制代码
#include <iostream>
#include <thread>
#include <vector>
#include <mutex>

std::vector<int> data;
std::mutex mtx;

void task() {
    for (int i = 0; i < 10000; ++i) {
        std::lock_guard<std::mutex> lock(mtx);
        data.push_back(i);
    }
}

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

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

    std::cout << data.size() << std::endl;
}

输出:

cpp 复制代码
20000

稳定。

九、锁的粒度问题(工程重点)

你这样写:

cpp 复制代码
void task() {
    std::lock_guard<std::mutex> lock(mtx);
    for (int i = 0; i < 10000; ++i) {
        data.push_back(i);
    }
}

锁范围更大。

优点:

  • 锁竞争少

缺点:

  • 阻塞时间更长

工程原则:

锁只保护"共享资源操作",

不要把耗时操作放在锁里。

十、错误示例:锁太小

cpp 复制代码
if (!data.empty()) {
    std::lock_guard<std::mutex> lock(mtx);
    data.pop_back();
}

问题:

data.empty() 没加锁。

可能:

线程 A 判断非空

线程 B pop 完

线程 A 再 pop → 崩溃

正确写法:

cpp 复制代码
std::lock_guard<std::mutex> lock(mtx);
if (!data.empty()) {
    data.pop_back();
}

十一、资源保护系统思维

写并发代码前问:

1️⃣ 这个数据是否共享?

2️⃣ 是否有写操作?

3️⃣ 锁由谁持有?

4️⃣ 锁范围是否最小化?

十二、死锁风险

死锁发生在:

  • 两个线程
  • 持有不同锁
  • 相互等待

例如:

线程 A:锁 m1 → 锁 m2

线程 B:锁 m2 → 锁 m1

工程规避方式:

统一加锁顺序。

十三、工程 Checklist

✅ 共享数据必须有唯一 mutex

✅ 永远使用 lock_guard(不要手动 lock/unlock)

✅ 判断 + 操作必须在同一锁内

✅ 锁只保护数据,不保护耗时操作


十四、Java 再类比总结

概念 Java C++
自动释放锁 synchronized lock_guard
手动锁 ReentrantLock std::mutex
异常安全 JVM 保证 RAII 保证

十五、本篇总结一句话

线程私有栈,共享堆;

共享必保护;

RAII 是 C++ 锁的灵魂。

下一篇预告

第三篇我们讲:

std::unique_lock ------ 为什么它比 lock_guard 更"重"?

  • 为什么条件变量必须用 unique_lock?
  • 如何避免死锁?
  • defer_lock / try_lock 是干什么的?
  • 锁释放与再获取的工程场景
相关推荐
秋916 小时前
Go语言(Golang)开发工程师全景解析:岗位职责·语言优势与使用场景·各城市薪资·发展前景·高考志愿填报(2026版)
开发语言·golang·高考
huangdong_17 小时前
1688商品图片采集技术解析:登录态处理与SKU图自动分类
开发语言
马士兵教育17 小时前
Java还有前景吗?Java+AI大模型学习路线及项目?
java·人工智能·python·学习·机器学习
搬砖魁首17 小时前
基础能力系列 - 多线程2 - 条件变量
c++·rust·条件变量·原子类型·线程同步互斥
chase_my_dream17 小时前
C++ + SLAM 高频面试问题整理
开发语言·c++·面试
snow@li17 小时前
Java:理解 Gradle / 后端项目的管家 / 打包SpringBoot 应用 / 完成编译、下载依赖、运行测试、打包 JAR/WAR / 速查表
java
牛油果子哥q17 小时前
【C++ STL string 】C++ STL string 终极精讲:底层原理、内存机制、全套API、深浅拷贝、易错坑点与工程实战规范
数据库·c++
Cloud_Shy61817 小时前
解读《Effective Python 3rd Edition》:从练气到老魔(第五章 Item 30 - 32)
开发语言·人工智能·笔记·python·学习方法
云烟成雨TD17 小时前
Spring AI 1.x 系列【57】动态工具发现:Tool Search Tool
java·人工智能·spring
zfoo-framework18 小时前
[修改代码使用]codex官方app中使用中转(不需要cc-switch) 1.config.toml 2.sk方式登录
java