实例分析: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 是什么,为什么需要这个将在后文中进行讲解。

相关推荐
楼田莉子2 天前
C++20新特性:Range库
开发语言·c++·后端·学习·c++20
楼田莉子3 天前
C++20现代特性:概念与约束
开发语言·c++·后端·学习·c++20
aluluka3 天前
C++ 20 协程的探索
c++·c++20
君鼎7 天前
内存池完整实现——C++20版
c++20·内存池
普通网友16 天前
记录我适配iOS26遇到的一些问题
c++20
前进吧-程序员16 天前
C++20/23 Ranges:从「迭代器对」到「可组合管道」
c++20
charlie11451419121 天前
基于开源项目的现代C++工程实践——OnceCallback 前置知识(下):C++20/23 高级特性
c++·开源·c++20
Hical_W21 天前
Hical 踩坑实录五部曲(二):MSVC / GCC / Clang 三平台 C++20 编译差异
linux·windows·经验分享·嵌入式硬件·macos·开源·c++20
Shan120523 天前
C++20中带有约束条件的new
c++20