0

0

C++多线程使用_C++标准库多线程组件应用教程

月夜之吻

月夜之吻

发布时间:2026-01-23 17:00:02

|

453人浏览过

|

来源于php中文网

原创

标准库多线程组件需谨慎使用:std::thread析构前必须join或detach,否则terminate;std::mutex须用lock_guard等RAII封装保证异常安全;std::atomic不保证复合操作原子性;std::async默认可能延迟执行,需显式指定launch::async确保并发。

c++多线程使用_c++标准库多线程组件应用教程

标准库std::threadstd::mutexstd::atomicstd::async 足够应对绝大多数多线程场景,但直接裸用容易出错——尤其是资源释放顺序、异常安全和竞态边界没理清时。

std::thread 构造后必须显式管理生命周期

构造 std::thread 对象后,它就拥有了一个关联的执行线程。如果该对象在析构前既未调用 join() 也未调用 detach(),程序会直接调用 std::terminate() 终止,且不抛异常、不打印堆

常见错误写法:

void worker() { /* ... */ }
void bad_example() {
    std::thread t{worker};  // 构造成功
    // 忘记 join/detach → 析构时 terminate!
}

正确做法:

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

  • 明确线程归属:需要等待结果 → 用 join();后台长期运行 → 用 detach()(但要确保所捕获变量的生命周期足够长)
  • 推荐封装:用 RAII 类(如自定义 scoped_thread)强制绑定 join()detach()
  • 避免临时对象绑定:不要写 std::thread{worker}().join(); —— 临时对象在语句末就析构,join() 调用无效

std::mutex 不能复制,且 lock/unlock 必须成对

std::mutex 是非可复制、非可移动类型,只能通过引用或指针共享。手动调用 lock()unlock() 极易遗漏(尤其在有异常分支的函数中)。

典型问题:

  • 忘记 unlock() 导致死锁
  • 异常路径跳过 unlock(),后续线程永远阻塞
  • 作用域重复 lock()(同一线程对同一 mutex 多次加锁 → 未定义行为)

务必使用 RAII 封装:

std::mutex mtx;
void safe_access() {
    std::lock_guard lk{mtx};  // 构造即 lock,析构即 unlock
    // 即使这里 throw 异常,lk 析构仍会 unlock
    do_something();
}

注意:std::lock_guard 不支持递归加锁;如需同一线程多次进入临界区,改用 std::recursive_mutex + std::lock_guard

Adobe Flex 简介 中文WORD版
Adobe Flex 简介 中文WORD版

Flex是一个基于组件的开发框架,可以生成一个由Flash Player运行的富互联网应用程序。Flex将基于标准的语言和各种可扩展用户界面及数据访问组件结合起来,使得开发人员能够构建具有丰富数据演示、强大客户端逻辑和集成多媒体的应用程序。 Flex是一个建立在Flash平台上的富客户端应用开发工具包,Flex 作为富 Internet 应用(RIA)时代的新技术代表,自从 2007 年 Adobe 公司将其开源以来,Flex 就以前所未有的速度在成长。感兴趣的朋友可以过来看看

下载

共享变量读写必须同步,atomic 不等于万能

即使只读或只写单个内置类型(如 int),只要多个线程访问同一对象,且至少一个为写操作,就必须同步。编译器重排、CPU 乱序、缓存不一致都可能导致未定义行为。

std::atomic 可解决简单标量的无锁读写,但要注意:

  • 默认内存序是 std::memory_order_seq_cst,性能开销略高;高频场景可降级(如用 relaxed 做计数器,acquire/release 做同步点)
  • std::atomic 不保护复合操作:比如 counter++ 是读-改-写三步,必须用 fetch_add() 等原子操作,而非普通赋值
  • 结构体/类不能直接 atomic 化,除非是 trivially copyable 且满足对齐等限制;否则仍需 mutex 保护

std::async 默认可能不启动新线程

std::async 的启动策略由 std::launch 参数控制,默认是 std::launch::deferred | std::launch::async,意味着实现可选择延迟执行(类似惰性求值),直到你调用 get()wait() 才真正运行 —— 这不是 bug,是标准允许的行为。

后果:

  • 你以为并发执行,实际是串行延迟执行
  • 若只创建 std::future 但不调用 get()/wait(),延迟任务永远不会跑

确保并发执行的方法:

auto f = std::async(std::launch::async, []{ return heavy_work(); });  // 显式指定 async
// 此时线程立即启动,不依赖 get() 触发

另外注意:std::future 析构时若未取值,会阻塞等待完成 —— 这容易导致意外的主线程挂起,尤其在容器里存了一堆 future 却忘了遍历 get()

多线程最难的从来不是“怎么启动线程”,而是“哪些数据被谁在什么时刻访问”。标准库组件只是工具,真正决定正确性的,是你对共享状态边界的判断精度,以及对每处 lockatomicjoin 背后隐含约束的清醒认知。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

197

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

190

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

358

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

542

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

53

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

393

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

574

2023.08.10

c++空格相关教程合集
c++空格相关教程合集

本专题整合了c++空格相关教程,阅读专题下面的文章了解更多详细内容。

0

2026.01.23

热门下载

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

精品课程

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

共61课时 | 3.5万人学习

React 教程
React 教程

共58课时 | 4万人学习

Pandas 教程
Pandas 教程

共15课时 | 1.0万人学习

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

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