多线程/std::thread线程退出方式详解

概述

这里默认你已经了解 std::thread 类的基本使用,和WinAPI多线程编程中 “如何优雅的退出线程” 等相关知识。阅读该文前,建议先看看《多线程 /C++ 11 std::thread 类深入理解和应用实践》《多线程/WinAPI线程退出方式比较分析》这两篇文章。在 函数 join函数 detach 的帮助文档中都讲到,
join(), After a call to this function, the thread object becomes non-joinable and can be destroyed safely.
detach(), After a call to std::thread::detach, the thread object becomes non-joinable and can be destroyed safely.
即在线程对象上调用 join函数或detach函数 后,线程对象便可安全地销毁了,何为安全,真的安全吗?

另外一个问题是,std::thread并未提供,像ExitThread、TerminateThread这样的终止线程的接口,也没有直接提供WaitForSingleObject 类似的等待函数, 当使用 std::thread 进行多线程编程时,似乎只有 “入口函数返回” 这一种方式。另外就是进程退出倒逼线程退出的方式。

不 join 也不 detach

int main()
{
    int interval_child = 1, interval_main = 2;  //s
    //
    std::thread t1 = std::thread([&]() {
        std::this_thread::sleep_for(std::chrono::seconds(interval_child));
        printf("at:%f, t1_entry_func return \n", DalOsTimeSysGetTime());
    });

    //预留时间,等待次线程结束
    std::this_thread::sleep_for(std::chrono::seconds(interval_main));
    //
    printf("at:%f, main end sleep t1.joinable:%d \n", DalOsTimeSysGetTime(), t1.joinable());

    //try {
    //    t1.join();
    //    printf("at:%f, t1.join return \n", DalOsTimeSysGetTime());
    //}
    //catch (const std::exception&) {
    //    std::cout << "any system_error exception \n";
    //}

    //system("pause");  
    
    return 0;
}

上述代码,运行结果如下:
在这里插入图片描述
可以得出,
1、即使线程对象代表的执行线程已经(函数返回)完成,此时对象 t1 依然是 joinable,可加入到其他线程中的。
2、进程退出前,如果没有对 std::thread 对象 t1 执行 join 或 detach 操作,则会触发abort终止进程。
补充,
一般情况下,无法通过异常处理机制捕获导致abort()调用的异常。当调用abort()函数时,理论上,可以通过一些系统相关的底层注册机制拦截或过滤它,我尝试了几种方式都没有成功,这里不再深究。

执行了detach并不能万事大吉

改动detach帮助下的示例程序如下,主要目的在于测试:进程退出后,子线程是否会 继续 完成执行过程。

#include <iostream>       // std::cout
#include <thread>         // std::thread, std::this_thread::sleep_for
#include <chrono>         // std::chrono::seconds
#include <string.h>
#include <fstream>
#include <stdio.h>
#include <windows.h>

using namespace std;

//辅助函数 //写log函数
void WriteLog(const char * format, ...)
{
    char buff[128] = { 0 };
    va_list ap;
    va_start(ap, format);
    vsprintf_s(buff, 128, format, ap);
    va_end(ap);
    //
    std::ofstream outfile("result.txt", std::ios_base::app);
    outfile.write(buff, strlen(buff));
    outfile.close();
}

//辅助函数 //时刻值ms //%f
double DalOsTimeSysGetTime(void)
{
    LARGE_INTEGER nFreq; LARGE_INTEGER nBeginTime;
    QueryPerformanceFrequency(&nFreq);
    QueryPerformanceCounter(&nBeginTime);
    return (double)(nBeginTime.QuadPart * 1000 / (double)nFreq.QuadPart);
}

//定义一个C++类对象
class ClassA
{
public:
    ClassA(int id): m_id(id) {
        WriteLog("CppObject-%d 构造 at %f\n", m_id, DalOsTimeSysGetTime());
        //申请堆内存
        m_pData = new TData();
    }
    ~ClassA() {
        WriteLog("CppObject-%d 析构 at %f\n", m_id, DalOsTimeSysGetTime());
        //释放堆内存
        if (NULL != m_pData) {
            delete m_pData;
            m_pData = NULL;
        }
    }

private:
    struct TData { //数据类
        int a;  int b;
    };

    TData *m_pData = NULL; int m_id = 0;
};

//入口函数
void pause_thread(int n)
{
    //定义在线程栈上的对象
    ClassA aObjectInStack(n);
    //使得指定的次线程睡眠n秒
    std::this_thread::sleep_for(std::chrono::seconds(n));
    //
    WriteLog("thread_%d pause %d seconds, then return at %f \n", n, n, DalOsTimeSysGetTime());
}

int main()
{
    std::ofstream outfile("result.txt", std::ios_base::trunc);
    outfile.close(); //清空日志文件

    std::cout << "Spawning and detaching 3 threads...\n";
    std::thread(pause_thread, 1).detach();
    std::thread(pause_thread, 2).detach();
    std::thread(pause_thread, 4).detach();
    std::cout << "Done spawning threads.\n";

    std::cout << "the main thread will now pause for 3 seconds\n";
    //you can give the detached threads time to finish (but not guaranteed!):
    std::this_thread::sleep_for(std::chrono::seconds(3));

    return 0;
}

//CppObject - 1 构造 at 112975428.942300
//CppObject - 2 构造 at 112975430.304200
//CppObject - 4 构造 at 112975431.947500
//thread_1 pause 1 seconds, then return at 112976440.094500
//CppObject - 1 析构 at 112976441.718200
//thread_2 pause 2 seconds, then return at 112977441.448000
//CppObject - 2 析构 at 112977442.140000
//id==4的线程 入口函数未完成执行,其内的对象未触发析构。

可以看出来,在进程退出后,
1、正在运行的线程入口函数的执行是"戛然而止"的,如果此时pause_thread入口函数内是while循环模式的,则线程将极有可能要死在while单次循环执行过程中,这是危险不优雅的。
2、入口函数若没有执行返回,则不会触发线程函数内对象的析构过程。
3、在 detach 作用下,并不会保证入口函数返回。 进程退出时,如果入口函数已经完成,则没有任何问题。但如果此时入口函数尚在执行过程中(如等待、耗时IO操作等),将与windowsAPI::ExitThread 的使用效果如出一辙。故,在使用detach分离线程后,若不加以控制使得入口函数能保证是可返回的,虽然系统会释放线程栈,但是由于此时析构过程未触发,依然存在m_pData堆内存泄漏的问题。

进程退出,
只要进程不退出(如将主线程使用system(“pause”) 或者 使用while循环睡眠,来保持运行),则detach的次线程便可以一直运行下去(如果它能一直运行下去),这是毋庸置疑的。如果宿主线程(或称MSDN中的calling thread调用线程)不是主线程,而是其他次线程,则线程对象在被detach后更不会退出执行。

建议使用 join 函数

讨论来讨论去,还是建议使用 join函数老实的等待执行线程退出。为此,我们可能需要将线程对象创建为成员变量或全局变量,以能在线程停止函数中调用join函数,实现等待操作。如下:

//在stop函数中利用join等待
void MyThreadStop() {
	m_runningFlag = fale;
	join();
}

平日里我们见到的示例程序大都简单到只是在main函数中创建子线程并直接退出,在main函数中 “同步地” 调用detach或join就完犊了。而实际使用中通常会更 “异步” 一些。如,在Qt环境的事件循环机制下,join的 “异步” 调用可能是:

#mian.cpp

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MianWnd w;
    w.show();
    return a.exec();
}

#mianWnd.h

class MianWnd : public QMainWindow
{
    Q_OBJECT
public:
    MianWnd(QWidget *parent = Q_NULLPTR);
    ~MianWnd();
private:
    //函数入口
    void pause_thread(int n);
    //停止函数
    void MyThreadStop();
private:
    //线程
    std::thread m_thread;
    //运行标志
    bool m_runningFlag = true;
    //堆栈资源
    struct TStruct
    { int a; int b; } *m_pResource;
private:
    Ui::MianWndClass ui;
};

#mainWnd.cpp

//辅助函数 //时刻值ms 
double DalOsTimeSysGetTime(void)
{
    LARGE_INTEGER nFreq; LARGE_INTEGER nBeginTime;

    QueryPerformanceFrequency(&nFreq);
    QueryPerformanceCounter(&nBeginTime);

    return (double)(nBeginTime.QuadPart * 1000 / (double)nFreq.QuadPart);
}

//辅助函数
void TraceForVStudio(char *fmt, ...)
{
    char out[1024] = { 0 };
    va_list body;
    va_start(body, fmt);
    vsprintf_s(out, 1024, fmt, body);
    va_end(body);    
    OutputDebugStringA(out); 
    OutputDebugStringA("\r\n");
}
//入口函数
void MianWnd::pause_thread(int n)
{
    while (m_runningFlag) //
    {
        if (NULL != m_pResource) //子线程内使用(共享)资源
            TraceForVStudio("Using Resource# a:%d b:%d ", ++m_pResource->a, ++m_pResource->b);

        std::this_thread::sleep_for(std::chrono::seconds(n));
    }
    //may do something other..
}
//停止函数
void MianWnd::MyThreadStop()
{
    m_runningFlag = false;
    m_thread.join();  //block
}
//构造函数
MianWnd::MianWnd(QWidget *parent)
    : QMainWindow(parent)
{
    ui.setupUi(this);
	//资源
    m_pResource = new TStruct();
    //线程
    m_thread = std::thread(&MianWnd::pause_thread, this, 5);
}
//析构函数
MianWnd::~MianWnd()
{
    //停止线程
    TraceForVStudio("Wait Begin At:%f", DalOsTimeSysGetTime()) ;
    MyThreadStop();
    TraceForVStudio("Wait Finish At:%f", DalOsTimeSysGetTime()); 
    //销毁线程资源
    if (NULL != m_pResource)
    { delete m_pResource; m_pResource = nullptr; }
    //销毁UI及其子窗口对象..
}

//关闭窗口触发析构过程
//Using Resource# a:1 b:1
//Using Resource# a:2 b:2
//...
//Wait Begin  At : 93813551.843300
//Exit Thread At : 93816981.917300 //about 3.5s
//Wait Finish At : 93816988.057200 //about 007ms

通过上述测试,可以确定join函数可以起到很好的等待线程退出的效果,比std::condition_variable 方便的多。上述,每5s完成单次循环,我随机关闭窗口触发析构过程,m_runningFlag 置零后大约过了3.5s后生效,然后入口函数退出,又过了7ms左右,join函数从阻塞过程中返回,析构过程继续执行堆栈资源的销毁过程。

<think>好的,用户之前已经询问过std::thread的基本用法,现在想了解如何用std::thread创建和管理多个线程。我需要结合之前的回答和提供的四个引用来组织内容。首先,用户的问题是如何开辟多个线程,所以需要展示多个线程的创建方法。引用3中的示例代码创建了两个线程t1和t2,这是一个很好的例子,应该包含进来。同时,引用2中的基本用法也提到了join方法,需要强调每个线程都需要正确管理。接下来,用户可能需要知道如何传递参数给多个线程,引用3中的例子传递了不同的参数给每个线程,这可以作为示例。此外,引用4中关于线程ID的获取,可以补充说明如何在线程函数中获取当前线程的ID,帮助用户区分不同线程。然后,需要考虑线程的管理,比如使用join还是detach。之前的回答提到过detach的风险,这里需要再次提醒用户,尤其是在多个线程的情况下,确保资源不被提前释放。另外,用户可能需要线程同步的提示,比如使用互斥锁来防止数据竞争。虽然引用中没有直接提到mutex,但这是管理多个线程时的常见需求,可以简要提及,并建议参考相关问题中的mutex用法。 还需要注意代码示例的完整性和正确性,比如正确的头文件包含和编译命令。引用2和3中的编译指令提到需要链接pthread库,这一点在多个线程的情况下同样适用,应该强调。最后,生成相关问题时要结合用户之前的提问,比如异常处理、线程同步、future和promise的区别,以及线程池的实现,这些都是与多线程管理紧密相关的话题,可以引导用户进一步学习。</think>### C++ `std::thread` 多线程创建与管理详解 使用 `std::thread` 创建和管理多个线程时,需重点关注**线程生命周期管理**、**参数传递**和**资源同步**。以下是具体实现方法及示例: --- #### **一、创建多个线程** 通过实例化多个 `std::thread` 对象实现并行任务: ```cpp #include <iostream> #include <thread> void task(int thread_id) { std::cout << "线程 " << thread_id << " 运行中" << std::endl; } int main() { // 创建3个线程 std::thread t1(task, 1); std::thread t2(task, 2); std::thread t3(task, 3); std::cout << "主线程继续执行..." << std::endl; // 等待所有子线程结束 t1.join(); t2.join(); t3.join(); std::cout << "所有子线程已完成" << std::endl; return 0; } ``` - **关键点**: 1. 每个线程对象必须调用 `join()` 或 `detach()`[^3] 2. 线程函数参数支持值传递和引用传递(需用 `std::ref`)[^3] --- #### **二、使用容器管理线程** 批量创建线程时,推荐使用容器(如 `vector`)统一管理: ```cpp #include <vector> int main() { std::vector<std::thread> threads; // 创建10个线程 for (int i = 0; i < 10; ++i) { threads.emplace_back([i] { std::cout << "线程 " << i << " 启动" << std::endl; }); } // 统一等待所有线程 for (auto& t : threads) { t.join(); } return 0; } ``` - **优势**:避免手动管理多个线程对象,简化代码逻辑[^2] --- #### **三、线程标识与调试** 通过 `std::thread::id` 区分不同线程: ```cpp void worker() { std::cout << "当前线程ID: " << std::this_thread::get_id() << std::endl; } int main() { std::thread t1(worker); std::thread t2(worker); std::cout << "主线程ID: " << std::this_thread::get_id() << std::endl; t1.join(); t2.join(); return 0; } ``` 输出示例: ``` 主线程ID: 140736245890944 当前线程ID: 123145447239680 当前线程ID: 123145452494848 ``` - **应用场景**:调试多线程竞争问题[^4] --- #### **四、注意事项** 1. **编译命令**:需链接 `pthread` 库 ```bash g++ -std=c++11 -pthread main.cpp -o main ``` 2. **资源竞争**:多个线程访问共享数据时需使用 `std::mutex` 同步 3. **异常安全**:在 `try-catch` 块中管理线程,避免未处理的异常导致资源泄漏 --- ### **总结与建议** - **优先选择 `join()`**:确保线程安全结束 - **避免全局变量**:通过参数传递数据 - **性能优化**:线程数量不宜超过CPU核心数,避免过度切换 ---
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值