首页 > 编程语言 >[转]32th@C++ 20新特性之线程与jthread@20240617

[转]32th@C++ 20新特性之线程与jthread@20240617

时间:2024-06-17 12:03:39浏览次数:28  
标签:std 20 thread 20240617 stop token 32th 线程 jthread

C++ 20新特性之线程与jthread

为什么要引入jthread

在C++ 11中,已经引入了std::thread。std::thread为C++标准库带来了一流的线程支持,极大地促进了多线程开发的便利性。但std::thread也存在一些明显的不足和短板,主要有以下几点。

1、生命周期管理的复杂性。std::thread对象必须在它代表的线程结束之前,一直保持存活。如果一个std::thread对象被销毁(比如:离开了其作用域),而它关联的线程还在运行,那么程序会调用std::terminate()终止,除非线程被join或者detach过。这就要求我们必须仔细管理每个线程对象的生命周期,大大增加了编码的复杂度和出错的可能性。

2、缺乏自动资源管理。std::thread没有自动管理线程生命周期的机制,程序员必须显式调用join或detach。否则,可能导致资源泄露或程序异常终止,这在异常处理场景下尤为麻烦。

3、异常安全性问题。如果在创建或启动std::thread时发生异常,可能会导致资源泄露或者程序的不确定行为。比如:如果在std::thread构造函数中抛出了异常,那么已经创建的线程可能无法正确地被join或detach。

为了解决这些问题,C++ 20中引入了std::jthread。

 

自动管理生命周期

C++ 20中新引入的std::jthread解决了C++ 11中std::thread的一些不便之处,特别是在线程生命周期管理上的自动化处理。std::jthread是一个智能指针风格的类,它自动join或detach与之关联的线程,从而避免了潜在的资源泄露问题。

接下来,我们通过一个具体的例子来理解std::jthread的工作原理。

#include <iostream>
#include <thread>
using namespace std;

void RunTask(stop_token stoken)
{
    int nCount = 0;
    while (!stoken.stop_requested())
    {
        cout << "Task running... " << nCount++ << endl;
        this_thread::sleep_for(chrono::seconds(1));
    }

    cout << "Task stopped" << endl;
}

int main()
{
    jthread t(RunTask);
    // 主线程等待一段时间
    this_thread::sleep_for(chrono::seconds(5));
    return 0;
}

在上面的示例代码中,RunTask函数作为工作线程的入口点,接收一个std::stop_token参数,用于检测是否请求停止。std::jthread t(RunTask)声明了一个jthread对象t,它会自动管理task函数所在线程的生命周期。当main函数结束时,t会自动调用join,等待关联线程完成或终止。可以看到,虽然我们没有显式要求停止线程,但当main函数返回时,jthread会确保线程安全结束。执行这段代码,其输出如下。

Task running... 0
Task running... 1
Task running... 2
Task running... 3
Task running... 4
Task stopped

 

stop_source和stop_token

与std::thread相比,std::jthread的强大之处在于它与std::stop_source和std::stop_token的集成,从而允许我们优雅地请求线程停止。

在下面的示例代码中,通过创建std::stop_source对象,并将其get_token方法的结果传递给RunTask函数,我们可以在需要时通过stopSource.request_stop()请求线程停止。RunTask函数中会循环检查std::stop_token的状态,一旦请求停止,就会退出循环并清理资源。

#include <iostream>
#include <thread>
using namespace std;

void RunTask(stop_token stoken)
{
    int nCount = 0;
    while (!stoken.stop_requested())
    {
        // 子线程执行一些任务
        cout << "Working..." << nCount++ << endl;
        this_thread::sleep_for(chrono::seconds(1));
    }

    cout << "Task stopped" << endl;
}

int main()
{
    stop_source stopSource;
    jthread t(RunTask, stopSource.get_token());

    // 主线程等待一段时间
    this_thread::sleep_for(chrono::seconds(3));
    cout << "Request task to stop..." << endl;

    // 主动请求线程停止
    stopSource.request_stop();
    return 0;
}

执行上述代码,其输出如下。

Working...0
Working...1
Working...2
Request task to stop...
Task stopped

 

线程中使用成员函数

std::jthread不仅可以用来启动普通函数,还可以用来启动类的成员函数。此时,需要使用lambda表达式来传递对象实例和成员函数指针。具体的用法,可以参考下面的示例代码。

#include <iostream>
#include <thread>
using namespace std;

class CTask
{
public:
    void Run(stop_token stoken)
    {
        int nCount = 0;
        while (!stoken.stop_requested())
        {
            cout << "Working..." << nCount++ << endl;
            this_thread::sleep_for(chrono::seconds(1));
        }
    }
};

int main()
{
    CTask task;
    jthread t([&task](stop_token stoken){ task.Run(stoken); });
    this_thread::sleep_for(chrono::seconds(5));
    return 0;
}

在上面的示例代码中,我们首先定义了一个名为CTask的类,其中包含一个公共成员函数Run。这个函数接收一个stop_token参数,用于检查是否有停止线程的请求。函数内部,它使用一个循环不断地输出计数器的值,并在每次循环之间暂停1秒。当stop_token表示停止请求时,循环结束。

在main函数中,我们创建了CTask类的对象task。接着,声明了一个jthread对象t,并初始化它以执行一个Lambda函数。这个Lambda函数捕获了task对象的引用,并将其传递给task.Run()方法,同时也传入了stop_token。jthread会自动为这个Lambda函数提供一个与之关联的stop_token,用于线程的停止请求。当jthread对象t的生命周期结束时,它会自动调用join来等待线程结束,无需手动调用join或detach。

标签:std,20,thread,20240617,stop,token,32th,线程,jthread
From: https://www.cnblogs.com/reformatio/p/18252100

相关文章

  • 2024年互联网技术与商务智能国际会议(ICITBI 2024)
    2024InternationalConferenceonInternetTechnologyandBusinessIntelligence【1】大会信息会议简称:ICITBI2024大会时间:2024-07-09大会地点:中国·三亚截稿时间:2024-06-25(以官网为准)审稿通知:投稿后2-3日内通知会议官网:www.iacitbi.com投稿邮箱:iacitbi@sub-pape......
  • GE WESDAC D20ME 模拟量扩展模块
    WESDACD20ME规格:总线类型:PROFIBUSDP 传输速率:9600bps至12Mbps 节点数量:最大31个 处理能力:32位主处理器,能够执行复杂的控制逻辑和算法 输入输出通道:支持多种输入和输出通道,可以监测和控制多个设备或过程 通讯接口:提供多种通讯接口,以便与其他系统进行通信 WESDA......
  • GE WESCOM D200 VME 通用电源接口模块
    WESCOMD200VME规格:尺寸:6U,单插槽电源:+5V,+3.3V,-12V动态功耗:10W静态功耗:5W处理器:1.33GHzIntelAtomProcessor内存:2GBDDR2SDRAM通信接口:支持以太网、USB、串口等多种I/O接口软件工具:提供一套完整的软件工具和驱动程序,支持Linux和Windows操作系统WESCOMD200VME功......
  • 洛谷 B3867 [GESP202309 三级] 小杨的储蓄 题解
     题目题目-[GESP202309三级]小杨的储蓄-洛谷题目描述小杨共有 ......
  • CVPR2024 分割Segmentation相关论文37篇速览
    Paper1MFP:MakingFullUseofProbabilityMapsforInteractiveImageSegmentation摘要小结:最近的交互式分割算法中,将先前的概率图作为网络输入,以帮助当前分割轮次的预测。然而,尽管使用了先前的掩膜,概率图中包含的有用信息并没有很好地传播到当前预测中。在本文中,为......
  • 证照之星XE个人破解版2024最新证照之星百度网盘免费下载
    随着社会的进步,证件照的用处越加宽广,例如:、结婚证件照、驾驶证照片、毕业证照片等等;当工作人员每天面对如此庞大的照片数量时,都尝试着去网上下载证照之星破解版,企图能够快速且专业的处理掉这些照片,这边是不建议这样做的。对于一些不会用PS,又想自己做证件照的时候,证照之星是一......
  • 2024.6.16
    publicclassSparkSQL09_Source_Req{publicstaticvoidmain(String[]args){//TODO在编码前,设定Hadoop的访问用户System.setProperty("HADOOP_USER_NAME","atguigu");finalSparkSessionsparkSession=SparkSession......
  • 【国赛赛题详解】2024年数学建模国赛ABCDEF题(点个关注,后续会更新)
     您的点赞收藏是我继续更新的最大动力!一定要点击如下的蓝色字体链接,那是获取资料的入口!点击链接加入群聊【2024国赛资料合集】:http://qm.qq.com/cgi-bin/qm/qr?_wv=1027&k=eQt5WRIvc5-fogZRrrahAhbqDa2nKfW8&authKey=%2BqQfThTxNnhw5LGJFRIcneF8JXBj1ufd2K01UpKPrpcgkKDskF......
  • 【国赛赛题详解】2024年数学建模国赛ABCDEF题(点个关注,后续会更新)
    您的点赞收藏是我继续更新的最大动力!一定要点击如下的蓝色字体链接,那是获取资料的入口!点击链接加入群聊【2024国赛资料合集】:http://qm.qq.com/cgi-bin/qm/qr?_wv=1027&k=eQt5WRIvc5-fogZRrrahAhbqDa2nKfW8&authKey=%2BqQfThTxNnhw5LGJFRIcneF8JXBj1ufd2K01UpKPrpcgkKDskFkr......
  • 2024/6/22 中考游记加超级游记合集
    高考day-3前我有抑郁症。高考day-2高中放高考假,共计\(7\)天。实放\(0\)天,因为我们可以回初中。然后因为大家都抑郁了所以相比之下我抑郁症好了。刚回初中就是试卷大礼包。/fn/fn/fn甚至还要搬寝室。/fn/fn/fn然后搬寝室的时候拿不下一本书就先放着。中间搬完之后......