0

0

c++中如何使用std::async的多线程启动策略_c++并发编程指南【实例】

穿越時空

穿越時空

发布时间:2026-01-20 02:25:24

|

625人浏览过

|

来源于php中文网

原创

c++kquote>std::async 默认启动策略由实现决定,但主流标准库实际走 async 模式(立即派生线程),非 lazy;未调用 get()/wait() 时任务可能已运行,线程创建不可控。

c++中如何使用std::async的多线程启动策略_c++并发编程指南【实例】

std::async 默认启动策略是 lazy 还是 eager?

默认是 std::launch::deferred | std::launch::async,也就是“由实现决定”——但绝大多数主流标准库(libstdc++、libc++、MSVC STL)在默认调用 std::async(func) 时,**实际走的是 async 模式(立即派生线程)**,不是 lazy。这点常被误解为“惰性求值”,其实不是。

关键在于:只要没调用 future.get()future.wait(),任务可能还没开始执行;但线程**很可能已经创建并运行中了**——这取决于调度器和系统负载,不是你可控的延迟。

  • 不显式指定策略时,行为不可移植,不能假设为 deferred
  • 若想确保延迟执行,必须写成 std::async(std::launch::deferred, func)
  • 若想强制并发(且禁止 deferred),用 std::async(std::launch::async, func)

std::async 启动失败时会怎样?常见崩溃点在哪里?

最典型的情况是系统线程数超限或资源不足,此时 std::async(std::launch::async, ...) 可能抛出 std::system_error,错误码为 std::errc::resource_unavailable_try_againstd::errc::operation_not_permitted

注意:std::launch::deferred 永远不会启动线程,所以不会因资源失败;但调用 get() 时才真正执行,这时可能抛出函数内部异常(非系统级)。

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

Magic Eraser
Magic Eraser

AI移除图片中不想要的物体

下载
  • 未捕获 std::system_error 会导致程序 terminate
  • 多个 std::async 短时间内密集调用,容易触发资源限制(尤其在嵌入式或容器环境)
  • Linux 上可通过 /proc/sys/kernel/threads-max 查看上限,ulimit -u 限制进程级线程数

如何避免 std::async 的隐式等待和生命周期陷阱?

std::future 析构时,若状态为 deferred,会**同步阻塞等待执行完成**;若为 async,则仅释放资源,不等待——但很多人误以为所有情况都不阻塞,结果在作用域结束时卡住主线程。

更隐蔽的问题是:把 std::async 返回的临时 std::future 绑定到右值引用(如 auto&& f = std::async(...)),仍无法延长其生命周期;C++17 起,临时对象绑定到 const lvalue 引用才能延长,但 future 不是 const。

  • 务必显式保存 future 对象:用 auto f = std::async(...),而非直接调用后丢弃
  • 避免在函数返回前让 future 被析构(尤其 deferred 模式下)
  • 不要依赖“不保存 future 就不执行”——async 模式下任务早已启动,只是结果无人读取
auto f = std::async(std::launch::async, []{ 
    std::this_thread::sleep_for(2s); 
    return 42; 
});
// 此处任务已在后台运行
int result = f.get(); // 阻塞直到完成

std::async 和 std::thread 在线程管理上根本区别是什么?

std::async 管理的是「异步操作的值」,不是线程本身;它不提供 join/detach 接口,也不让你控制线程生命周期。而 std::thread 是对 OS 线程的轻量封装,必须显式管理(否则析构时 terminate)。

这意味着:std::async 更适合“发个任务,之后取结果”的场景;不适合需要长期驻留、反复通信、或精细控制线程状态(如暂停、优先级、亲和性)的场合。

  • std::async 无法设置线程名(Linux 用 pthread_setname_np,Windows 用 SetThreadDescription),std::thread 可以
  • std::async 不支持移动到另一个作用域(future 可移动,但执行上下文绑定在原 async 调用点)
  • 大量短任务用 std::async 易造成线程频繁创建销毁,开销大;应考虑线程池 + std::packaged_task
实际用的时候,最容易忽略的是 deferred 模式下 future 析构的隐式同步,以及 async 模式下无节制调用引发的系统级资源耗尽——这两点既不报编译错,也难在测试中暴露,上线后才突然卡死或崩溃。

相关专题

更多
c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

524

2023.09.20

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1024

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

66

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

450

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

9

2026.01.19

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

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

481

2023.08.10

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

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

143

2025.12.24

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

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

61

2025.12.01

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

23

2026.01.19

热门下载

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

精品课程

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

共48课时 | 7.4万人学习

Git 教程
Git 教程

共21课时 | 2.8万人学习

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

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