C語言中pthread_exit()函數(shù)實現(xiàn)終止線程的方法

本文講解"c語言中pthread_exit()函數(shù)實現(xiàn)終止線程的方法",希望能夠解決相關(guān)問題。

 

多線程編程中,線程結(jié)束執(zhí)行的方式有 3 種,分別是:

  • 線程將指定函數(shù)體中的代碼執(zhí)行完后自行結(jié)束;

  • 線程執(zhí)行過程中,被同一進程中的其它線程(包括主線程)強制終止;

  • 線程執(zhí)行過程中,遇到 pthread_exit() 函數(shù)結(jié)束執(zhí)行。

注意,默認屬性的線程執(zhí)行結(jié)束后并不會立即釋放占用的資源,直到整個進程執(zhí)行結(jié)束,所有線程的資源以及整個進程占用的資源才會被操作系統(tǒng)回收。
實現(xiàn)線程資源及時回收的常用方法有兩種,一種是修改線程屬性,另一種是在另一個線程中調(diào)用 pthread_join() 函數(shù),我們會在后續(xù)章節(jié)中給您詳細介紹這兩種方法。

線程結(jié)束執(zhí)行的 3 種方式中,第 1 種很容易理解,我們會在《pthread_cancel()函數(shù)》一文中介紹第 2 種方式,本文重點講解 pthread_exit() 函數(shù)的功能和用法。

pthread_exit()函數(shù)的用法

linux pthread_exit() 函數(shù)聲明在頭文件中,語法格式如下所示:

void pthread_exit(void *retval);

retval 是 void* 類型的指針,可以指向任何類型的數(shù)據(jù),它指向的數(shù)據(jù)將作為線程退出時的返回值。如果線程不需要返回任何數(shù)據(jù),將 retval 參數(shù)置為 null 即可。

注意,retval 指針不能指向函數(shù)內(nèi)部的局部數(shù)據(jù)(比如局部變量)。換句話說,pthread_exit() 函數(shù)不能返回一個指向局部數(shù)據(jù)的指針,否則很可能使程序運行結(jié)果出錯甚至崩潰。

接下來通過一個樣例,給大家演示 pthread_exit() 函數(shù)的用法(樣例一):

#include 
#include 
//線程要執(zhí)行的函數(shù),arg 用來接收線程傳遞過來的數(shù)據(jù)
void *threadfun(void *arg)
{
    //終止線程的執(zhí)行,將“http://c.biancheng.net”返回
    pthread_exit("http://c.biancheng.net"); //返回的字符串存儲在常量區(qū),并非當前線程的私有資源
    printf("*****************");//此語句不會被線程執(zhí)行
}
int main()
{
    int res;
    //創(chuàng)建一個空指針
    void * thread_result;
    //定義一個表示線程的變量
    pthread_t mythread;
    res = pthread_create(&mythread, null, threadfun, null);
    if (res != 0) {
        printf("線程創(chuàng)建失敗");
        return 0;
    }
    //等待 mythread 線程執(zhí)行完成,并用 thread_result 指針接收該線程的返回值
    res = pthread_join(mythread, &thread_result);
    if (res != 0) {
        printf("等待線程失敗");
    }
    printf("%s", (char*)thread_result);
    return 0;
}

假設程序存儲在 thread.c 文件中,執(zhí)行過程如下:

[root@localhost ~]# gcc thread.c -o thread.exe -lpthread
[root@localhost ~]# ./thread.exe
http://c.biancheng.net

通過執(zhí)行結(jié)果不難看出,mythread 線程并沒有執(zhí)行 threadfun() 函數(shù)中最后一個 printf() 語句,從側(cè)面驗證了 pthread_exit() 函數(shù)的功能。此外,我們通過在主線程(main() 函數(shù))調(diào)用 pthread_join() 函數(shù),獲取到了 mythread 線程返回的數(shù)據(jù)。有關(guān) pthread_join() 函數(shù)的功能和用法,我們會在《獲取線程函數(shù)返回值》一節(jié)中給大家講解。

pthread_exit() 和 return 的區(qū)別

如果想在線程執(zhí)行結(jié)束時返回指定的數(shù)據(jù),除了用 pthread_exit() 函數(shù)外,還可以使用 return 語句。

修改樣例一中的程序,將第 8 行(調(diào)用 pthread_exit() )代碼替換成如下語句:

return "http://yisu.com";

重新編譯、執(zhí)行此程序,會發(fā)現(xiàn)程序的執(zhí)行結(jié)果和之前完全相同。這意味著當線程執(zhí)行結(jié)束時,無論是采用 return 語句還是調(diào)用 pthread_exit() 函數(shù),主線程中的 pthread_join() 函數(shù)都可以接收到線程的返回值。

那么,return 語句和 pthread_exit() 函數(shù)的區(qū)別是什么呢?

首先,return 語句和 pthread_exit() 函數(shù)的含義不同,return 的含義是返回,它不僅可以用于線程執(zhí)行的函數(shù),普通函數(shù)也可以使用;pthread_exit() 函數(shù)的含義是線程退出,它專門用于結(jié)束某個線程的執(zhí)行。

在主線程(main() 函數(shù))中,return 和 pthread_exit() 函數(shù)的區(qū)別最明顯。舉個例子:

#include 
#include 
void *threadfun(void *arg)
{
    sleep(5);//等待一段時間
    printf("http://yisu.com\n");
}
int main()
{
    int res;
    pthread_t mythread;
    res = pthread_create(&mythread, null, threadfun, null);
    if (res != 0) {
        printf("線程創(chuàng)建失敗");
        return 0;
    }
    printf("億速云\n");
    return 0;
}
編譯、執(zhí)行此程序,輸出結(jié)果為:
億速云

通過執(zhí)行結(jié)果可以看到,主線程正常執(zhí)行結(jié)束,mythread 線程并沒有輸出指定的數(shù)據(jù)。原因很簡單,主線程執(zhí)行速度很快,主線程最后執(zhí)行的 return 語句不僅會終止主線程執(zhí)行,還會終止其它子線程執(zhí)行。也就是說,mythread 線程還沒有執(zhí)行輸出語句就被終止了。

將上面程序中,main() 函數(shù)中的return 0;用如下語句替換:

pthread_exit(null);

重新編譯、執(zhí)行程序,運行結(jié)果為:

億速云
http://yisu.com

對比上面兩個執(zhí)行結(jié)果,我們可以得出的結(jié)論是:pthread_exit() 函數(shù)只會終止當前線程,不會影響進程中其它線程的執(zhí)行。

此外,pthread_exit() 可以自動調(diào)用線程清理程序(本質(zhì)是一個由 pthread_cleanup_push() 指定的自定義函數(shù)),return 則不具備這個能力??傊趯嶋H場景中,如果想終止某個子線程執(zhí)行,強烈建議大家使用 pthread_exit() 函數(shù)。終止主線程時,return 和 pthread_exit() 函數(shù)發(fā)揮的功能不同,可以根據(jù)需要自行選擇。

相關(guān)文章
亚洲国产精品第一区二区,久久免费视频77,99V久久综合狠狠综合久久,国产免费久久九九免费视频