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

相关推荐
十五年专注C++开发5 天前
全面深入了解C++20 范围库(std::ranges)
c++20·管道·哨兵·视图·ranges
小小龙学IT8 天前
C++20 协程深度解析:从原理到高性能异步框架实战
junit·c++20
楼田莉子13 天前
C++20新特性:协程
开发语言·c++·后端·学习·c++20
ouliten15 天前
C++笔记:C++20风格线程池
c++·笔记·c++20
眠りたいです16 天前
现代C++:C++17中的新库特性
开发语言·c++·c++20·c++17
楼田莉子23 天前
C++20新特性:Range库
开发语言·c++·后端·学习·c++20
楼田莉子24 天前
C++20现代特性:概念与约束
开发语言·c++·后端·学习·c++20
aluluka24 天前
C++ 20 协程的探索
c++·c++20
君鼎1 个月前
内存池完整实现——C++20版
c++20·内存池
普通网友1 个月前
记录我适配iOS26遇到的一些问题
c++20