C++多线程编程:join与detach的致命陷阱

一、tha.jion 和 tha.deatch

1. 先讲两个函数的区别

tha.join();

  • 主线程等着子线程跑完,再继续往下走
  • 主线程会卡在这里,等子线程结束
  • 安全、不会出错

tha.detach();

  • 将子线程从主线程分离,主线程不再管理子线程,各自独立运行
  • 主线程继续往下跑,可能先跑完、直接退出程序
  • 子线程变成后台线程(守护线程),继续自己跑
  • 极容易出 bug!

2. 放到代码里,会发生什么灾难?

复制代码
Int a(10);
std::thread tha(funa, a);
tha.detach();  // 这里改成detach

// 主线程直接往下跑!不等子线程!
cout << "main---------------" << endl;
a.Print();
return 0;

结果 1:主线程可能先跑完,直接销毁 a

主线程退出 → 局部变量 a 被析构但子线程还在跑,还在使用已经被销毁的对象悬空对象、非法访问、程序直接崩溃!

结果 2:输出乱序

你会看到输出乱跳:

复制代码
main---------------
10
Creat Int:...
Copy Int:...

主线程先打印完了,子线程才开始跑。

结果 3:子线程还没跑完,程序已经结束

子线程直接被操作系统强行杀死

二、条件变量condition_variable

它就是 C++11 专门给线程用的 "等待 + 唤醒" 工具

1. 它到底是干嘛的?

一句话:让一个线程停下来等,直到另一个线程喊它,它才继续跑。

你可以把它理解成:

  • 线程 A:我等着,你弄好了叫我
  • 线程 B:我弄好了,喊你继续

这就是 condition_variable

2. 它必须配合谁一起用?

必须配合 std::mutex 互斥锁一起用不能单独用!

因为:

  • 等待的时候要解锁
  • 唤醒的时候要加锁条件变量自己不会锁,必须靠 mutex

3. 最核心的 3 个功能

  1. **wait()**线程停下来,阻塞等待→ 自动解锁 → 等着被唤醒

  2. notify_one() 唤醒一个正在等待的线程

  3. notify_all() 唤醒所有正在等待的线程

三、惊群现象

惊群现象 是指:使用 notify_all() 时,一次性唤醒所有等待线程,但只有一个线程能真正工作,其余线程白唤醒、白抢锁,造成 CPU 资源浪费。

在多线程轮流打印场景中:

  • notify_one 可能唤醒错误线程 → 死锁
  • notify_all 会产生惊群 → 但安全、不会死锁 所以必须用 notify_all

四、notify_one和notify_all

1.对比

notify_one()

  • 只唤醒1个正在 wait 的线程

  • 优点:无惊群,效率高

  • 缺点:可能唤醒 "错误线程" → 死锁

notify_all()

  • 唤醒所有正在 wait 的线程

  • 优点:绝对安全,不会死锁

  • 缺点:惊群效应,大量线程抢锁,浪费 CPU

2.什么时候用哪个?

用 notify_one 的场景:生产者消费者(一对多)

用 notify_all 的场景:多线程轮流打印 / 条件与线程绑定 / 程序退出

相关推荐
小樱花的樱花2 小时前
4 文件选择对话框 QFileDialog
开发语言·c++·ui
leaves falling2 小时前
深入浅出 C++ STL list:从入门到精通
开发语言·c++
史迪仔01123 小时前
[QML] 交互事件深度解析:鼠标、键盘、拖拽
前端·c++·qt
米啦啦.3 小时前
类继承、子类拷贝构造函数、赋值运算符重载函数、多继承(虚继承)
c++·多继承·类继承·赋值运算符重载
swift192213 小时前
Qt多语言问题 —— 静态成员变量
开发语言·c++·qt
用户805533698033 小时前
现代Qt开发教程(新手篇)1.4——容器
c++·qt
ulias2123 小时前
Linux中的开发工具
linux·运维·服务器·开发语言·c++·windows
qq_466302454 小时前
u盘插入拔出,listView不显示盘符变化
c++·qt
小熊Coding4 小时前
Windows 上安装 mysqlclient 时遇到了编译错误,核心原因是缺少 Microsoft Visual C++ 14.0 或更高版本 的编译环境。
c++·windows·python·microsoft·django·mysqlclient·bug记录