0

0

c++的std::jthread相比std::thread有哪些改进? (C++20自动join)

冰火之心

冰火之心

发布时间:2026-01-14 12:12:09

|

892人浏览过

|

来源于php中文网

原创

c++kquote>std::jthread 构造后自动管理生命周期,析构时默认尝试 join() 以避免 std::terminate();内置 std::stop_token 支持协作式中断;仅支持移动且移动后源对象状态明确;需 C++20 支持。

c++的std::jthread相比std::thread有哪些改进? (c++20自动join)

std::jthread 构造后自动管理生命周期,无需手动调用 join()detach()

std::jthread 在构造时绑定可调用对象,并在析构时自动尝试 join() —— 这是它最核心的改进。如果线程仍在运行且未被 join()detach(),析构会阻塞直到线程结束;若线程已不可加入(如已 detach()),则不执行任何操作。这直接消除了 std::thread 常见的“析构前未 join/detach 导致程序终止”的风险。

常见错误现象:std::thread 对象离开作用域时仍处于 joinable() 状态,会触发 std::terminate()。而 std::jthread 默认行为就是安全的。

  • 适用于短生命周期、作用域内启动并等待完成的线程场景(如函数局部线程)
  • 不适用于需要长期后台运行、且明确交由其他模块管理生命周期的线程(此时仍应显式 detach() 或移交所有权)
  • 若你确实想避免析构时阻塞,可提前调用 detach() —— 但需自行承担资源/生命周期责任

内置协作式中断支持:std::jthread 携带 std::stop_tokenstd::stop_source

std::jthread 内置一个 std::stop_source,其 get_stop_token() 可返回对应的 std::stop_token,供线程函数检查是否收到停止请求。这是 C++20 对“协作取消”的标准化支持,比手写 flag + 条件变量更轻量、更统一。

使用场景:需要响应外部信号提前退出的长时间运行任务(如轮询、IO 等待、计算循环)。

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

小鸽子助手
小鸽子助手

一款集成于WPS/Word的智能写作插件

下载
  • std::jthread 的构造函数可接受带 std::stop_token 参数的可调用对象(签名形如 void(std::stop_token)
  • 线程函数内部可通过 token.stop_requested() 检查,或用 token.stop_callback(...) 注册回调
  • 主线程可通过 jthread.request_stop() 发起中断请求,该操作是线程安全且无锁
  • 注意:中断只是“建议”,具体是否响应、何时响应,完全由线程函数逻辑决定

移动语义更安全,禁止拷贝,且移动后源对象变为不可连接状态

std::thread 类似,std::jthread 删除了拷贝构造和拷贝赋值,只支持移动。但关键区别在于:移动后,源 std::jthread 对象的状态被明确定义为“不可连接(not joinable)”,且其内部 stop_source 也被转移走 —— 不会残留无效句柄或悬空 token。

对比 std::thread:移动后源对象虽也变为 joinable() == false,但其内部状态未定义(仅保证不抛异常),而 std::jthread 明确保证移动后源对象的 get_stop_token() 返回空 std::stop_token,且 joinable()false

  • 适合封装在线程池、任务队列等需要转移线程所有权的场景
  • 移动后对原对象调用 request_stop()get_stop_token() 是安全的(返回空 token,不触发操作)
  • 仍需注意:移动后的目标对象才拥有真实资源,原对象不应再用于同步或控制

性能与兼容性:无额外运行时开销,但要求 C++20 编译器支持

std::jthread 的实现不引入额外的原子操作或内存分配,其 stop_source上对象,stop_token 是轻量 handle。自动 join() 行为只在析构时发生一次,与手写 ~T() { if (t.joinable()) t.join(); } 成本一致。

限制也很明确:必须启用 C++20(如 GCC 10+、Clang 12+、MSVC 19.29+),且标准库需完整实现 。部分旧版 libc++ 或 libstdc++ 可能缺失部分接口。

  • 编译需加 -std=c++20,链接无需额外库
  • 若需跨平台或支持较老工具链,不能假设 std::jthread 可用,应回退到 std::thread + RAII 封装
  • 目前没有运行时降级机制 —— 它不是 std::thread 的子类,也不能隐式转换
#include 
#include 
#include 

void worker(std::stop_token stoken) {
    for (int i = 0; i < 5; ++i) {
        if (stoken.stop_requested()) {
            std::cout << "worker: interrupted\n";
            return;
        }
        std::this_thread::sleep_for(100ms);
    }
    std::cout << "worker: done\n";
}

int main() {
    std::jthread t{worker};  // 自动管理生命周期 + 提供 stop_token
    std::this_thread::sleep_for(300ms);
    t.request_stop();  // 安全发起中断
    // t 析构时自动 join() —— 无需手动处理
}
真正容易被忽略的是:自动 join() 虽安全,但可能掩盖设计问题 —— 比如本该异步执行的任务被意外阻塞在析构,导致调用方延迟。协作中断也不是银弹,stop_token 无法中断系统调用或 mutex 等待,仍需结合超时、条件变量等手段。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

736

2023.08.22

登录token无效
登录token无效

登录token无效解决方法:1、检查token的有效期限,如果token已经过期,需要重新获取一个新的token;2、检查token的签名,如果签名不正确,需要重新获取一个新的token;3、检查密钥的正确性,如果密钥不正确,需要重新获取一个新的token;4、使用HTTPS协议传输token,建议使用HTTPS协议进行传输 ;5、使用双因素认证,双因素认证可以提高账户的安全性。

6083

2023.09.14

登录token无效怎么办
登录token无效怎么办

登录token无效的解决办法有检查Token是否过期、检查Token是否正确、检查Token是否被篡改、检查Token是否与用户匹配、清除缓存或Cookie、检查网络连接和服务器状态、重新登录或请求新的Token、联系技术支持或开发人员等。本专题为大家提供token相关的文章、下载、课程内容,供大家免费下载体验。

801

2023.09.14

token怎么获取
token怎么获取

获取token值的方法:1、小程序调用“wx.login()”获取 临时登录凭证code,并回传到开发者服务器;2、开发者服务器以code换取,用户唯一标识openid和会话密钥“session_key”。想了解更详细的内容,可以阅读本专题下面的文章。

1058

2023.12.21

token什么意思
token什么意思

token是一种用于表示用户权限、记录交易信息、支付虚拟货币的数字货币。可以用来在特定的网络上进行交易,用来购买或出售特定的虚拟货币,也可以用来支付特定的服务费用。想了解更多token什么意思的相关内容可以访问本专题下面的文章。

1213

2024.03.01

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

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

97

2025.11.27

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

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

1017

2023.10.19

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

2

2026.01.14

热门下载

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

精品课程

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

共58课时 | 3.6万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

ASP 教程
ASP 教程

共34课时 | 3.5万人学习

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

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