实例分析:C++20的std::jthread

我们首先开门见山来介绍一下,在C++20 的 std::jthread具有两大特性:

自动合并 joining thread

线程取消 stop token

其中 jthread 的 "j" 指的就是 joining。

自动合并 joining thread

std::jthread 的第一个特点就是能够再析构函数中自动调用 join() 从而避免了无法合并的问题。

析构的变化:

其实具体实现的思路与上文中手动 diy 实现的类似。下面是 msvc 的源码。这里的操作比上文编写的多了一步,_Ssource.request_stop(); 请求停止。其余功能一致。

复制代码
class {
    // pass
    
	~jthread() {
        _Try_cancel_and_join();
    }

    void _Try_cancel_and_join() noexcept {
        if (_Impl.joinable()) {
            _Ssource.request_stop();
            _Impl.join();
        }
    }
};

析构函数有了变化,那么构造函数也有变化吗?答案是确实有也变化。

构造的变化:

下面是 msvc 中 jthread 的有参构造函数。

复制代码
    template <class _Fn, class... _Args, enable_if_t<!is_same_v<remove_cvref_t<_Fn>, jthread>, int> = 0>
    _NODISCARD_CTOR explicit jthread(_Fn&& _Fx, _Args&&... _Ax) {
        if constexpr (is_invocable_v<decay_t<_Fn>, stop_token, decay_t<_Args>...>) {
            _Impl._Start(_STD forward<_Fn>(_Fx), _Ssource.get_token(), _STD forward<_Args>(_Ax)...);
        } else {
            _Impl._Start(_STD forward<_Fn>(_Fx), _STD forward<_Args>(_Ax)...);
        }
    }

在 if constexpr 中,会在编译期判断首个函数参数是否是 std::stop_token。基于这个条件的分支进行具体的线程创建。而 else 中就是在 C++11 中创建线程的常规方式。

而 std::stop_token 是什么,为什么需要这个将在后文中进行讲解。

相关推荐
charlie1145141912 小时前
基于开源项目的现代C++工程实践——OnceCallback 前置知识(下):C++20/23 高级特性
c++·开源·c++20
Hical_W16 小时前
Hical 踩坑实录五部曲(二):MSVC / GCC / Clang 三平台 C++20 编译差异
linux·windows·经验分享·嵌入式硬件·macos·开源·c++20
Shan12052 天前
C++20中带有约束条件的new
c++20
Hical_W5 天前
用 Hical + MySQL 5 分钟搭建 CRUD API(C++20 协程版)
数据库·mysql·c++20
Hical_W5 天前
从 io_context 出发,掌握 C++20 协程式异步 I/O,学会 TCP 服务器、定时器和多线程模型,结合 Hical 框架实战解读
服务器·tcp/ip·开源·c++20
c++之路10 天前
C++20概述
java·开发语言·c++20
故事还在继续吗11 天前
C++20关键特性
开发语言·c++·c++20
熊文豪12 天前
FinceptTerminal 深度解析:用 C++20 + Qt6 + Python 打造的开源 Bloomberg 终端
python·开源·c++20·bloomberg·finceptterminal
前进吧-程序员23 天前
现代 C++ 异步编程:从零实现一个高性能 ThreadPool (C++20 深度实践)
开发语言·c++·c++20