0

0

c++ multi-threading多线程编程_c++ thread库用法

冰火之心

冰火之心

发布时间:2026-01-12 07:37:33

|

172人浏览过

|

来源于php中文网

原创

std::thread构造后必须join或detach,否则析构时调用std::terminate终止程序;传参默认拷贝,引用需std::ref;共享数据须用mutex等同步机制避免data race。

c++ multi-threading多线程编程_c++ thread库用法

std::thread 构造后必须 detach 或 join,否则程序崩溃

构造 std::thread 对象后,若其处于“可加入”(joinable)状态而未调用 join()detach(),析构时会调用 std::terminate(),直接终止进程——这不是异常,无法捕获。

  • 常见错误:在函数内创建 std::thread 但忘记处理,尤其在提前 return 或异常路径中
  • 推荐做法:用 RAII 封装,比如自定义 scoped_thread,或立即 join()(适合需等待完成的场景)
  • detach() 后线程后台运行,与主线程生命周期解耦,但要注意避免访问已销毁对象(如局部变量、内存)
void do_work() { /* ... */ }
int main() {
    std::thread t{do_work};
    // ❌ 错误:t 析构前未 join/detach
    // ✅ 正确之一:t.join();
    // ✅ 正确之二:t.detach();
}

传递参数给 std::thread 时默认按值拷贝,引用需 wrap with std::ref

std::thread 构造函数对参数执行完美转发,但**所有参数默认按值传递并拷贝**。若想传引用,必须显式包装为 std::ref(x)std::cref(x);否则你在线程里操作的是副本,主函数中的原始变量不受影响。

  • 传指针安全,但需确保所指对象生命周期长于线程执行时间
  • 传 lambda 捕获引用同理:若用 [&] 捕获局部变量,线程可能访问已销毁栈帧
  • 移动语义可用:std::thread t{func, std::move(obj)};,此时原对象失效
void increment(int& x) { ++x; }
int main() {
    int a = 0;
    std::thread t{increment, std::ref(a)}; // ✅ 引用传递
    t.join();
    // a == 1
}

共享数据不加锁导致 data race,std::mutex 是最常用同步原语

多个线程同时读写同一内存位置(至少一个为写),且无同步机制,即构成 data race —— 行为未定义(可能 crash、静默错值、优化器乱序重排)。std::mutex 是最基础也最常用的互斥方案。

  • 务必成对使用 lock()/unlock(),推荐用 std::lock_guardstd::unique_lock 自动管理
  • 避免锁粒度太粗(性能差)或太细(易漏锁、死锁);优先保护最小必要临界区
  • 不要跨线程传递未加锁的裸指针/引用指向共享对象;考虑用 std::shared_ptr 配合 mutex 管理所有权和访问
#include 
std::mutex mtx;
int counter = 0;

void safe_increment() { std::lock_guard lock{mtx}; // 析构自动 unlock ++counter; }

std::async + std::future 更适合“启动即等待结果”的任务模型

当你要启动一个异步计算,并稍后获取返回值,std::async 比手动管理 std::thread 更简洁、更安全:它自动处理线程生命周期、异常传播和结果同步。

Ideogram
Ideogram

Ideogram是一个全新的文本转图像AI绘画生成平台,擅长于生成带有文本的图像,如LOGO上的字母、数字等。

下载

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

  • 默认策略是 std::launch::async | std::launch::deferred,编译器可选择延迟执行(调用 get() 时才运行),也可能立即启动新线程
  • 若强制立即并发,用 std::async(std::launch::async, func, args...)
  • std::future::get() 是阻塞调用,且只能调用一次;多次调用抛 std::future_error
auto fut = std::async([]{ return 42; });
int result = fut.get(); // 阻塞直到完成,返回 42

多线程最难的不是语法,而是判断哪些数据真正共享、谁负责生命周期、锁是否覆盖了所有访问路径——哪怕只漏一个读操作,data race 就可能在压力测试时才爆发。

相关专题

更多
lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

202

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

Python lambda详解
Python lambda详解

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

41

2026.01.05

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

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

386

2023.07.18

堆和栈区别
堆和栈区别

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

569

2023.08.10

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

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

479

2023.08.10

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

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

143

2025.12.24

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

60

2025.12.01

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

80

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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