C語言中pthread_exit()函數(shù)實現(xiàn)終止線程
多線程編程中,線程結束執(zhí)行的方式有 3 種,分別是:
- 線程將指定函數(shù)體中的代碼執(zhí)行完后自行結束;
- 線程執(zhí)行過程中,被同一進程中的其它線程(包括主線程)強制終止;
- 線程執(zhí)行過程中,遇到 pthread_exit() 函數(shù)結束執(zhí)行。
注意,默認屬性的線程執(zhí)行結束后并不會立即釋放占用的資源,直到整個進程執(zhí)行結束,所有線程的資源以及整個進程占用的資源才會被操作系統(tǒng)回收。
實現(xiàn)線程資源及時回收的常用方法有兩種,一種是修改線程屬性,另一種是在另一個線程中調用 pthread_join() 函數(shù),我們會在后續(xù)章節(jié)中給您詳細介紹這兩種方法。
線程結束執(zhí)行的 3 種方式中,第 1 種很容易理解,我們會在《pthread_cancel()函數(shù)》一文中介紹第 2 種方式,本文重點講解 pthread_exit() 函數(shù)的功能和用法。
pthread_exit()函數(shù)的用法
Linux pthread_exit() 函數(shù)聲明在<pthread.h>頭文件中,語法格式如下所示:
void pthread_exit(void *retval);
retval 是 void* 類型的指針,可以指向任何類型的數(shù)據(jù),它指向的數(shù)據(jù)將作為線程退出時的返回值。如果線程不需要返回任何數(shù)據(jù),將 retval 參數(shù)置為 NULL 即可。
注意,retval 指針不能指向函數(shù)內部的局部數(shù)據(jù)(比如局部變量)。換句話說,pthread_exit() 函數(shù)不能返回一個指向局部數(shù)據(jù)的指針,否則很可能使程序運行結果出錯甚至崩潰。
接下來通過一個樣例,給大家演示 pthread_exit() 函數(shù)的用法(樣例一):
#include <stdio.h> #include <pthread.h> //線程要執(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í)行結果不難看出,myThread 線程并沒有執(zhí)行 ThreadFun() 函數(shù)中最后一個 printf() 語句,從側面驗證了 pthread_exit() 函數(shù)的功能。此外,我們通過在主線程(main() 函數(shù))調用 pthread_join() 函數(shù),獲取到了 myThread 線程返回的數(shù)據(jù)。有關 pthread_join() 函數(shù)的功能和用法,我們會在《獲取線程函數(shù)返回值》一節(jié)中給大家講解。
pthread_exit() 和 return 的區(qū)別
如果想在線程執(zhí)行結束時返回指定的數(shù)據(jù),除了用 pthread_exit() 函數(shù)外,還可以使用 return 語句。
修改樣例一中的程序,將第 8 行(調用 pthread_exit() )代碼替換成如下語句:
return "http://jb51.net";
重新編譯、執(zhí)行此程序,會發(fā)現(xiàn)程序的執(zhí)行結果和之前完全相同。這意味著當線程執(zhí)行結束時,無論是采用 return 語句還是調用 pthread_exit() 函數(shù),主線程中的 pthread_join() 函數(shù)都可以接收到線程的返回值。
那么,return 語句和 pthread_exit() 函數(shù)的區(qū)別是什么呢?
首先,return 語句和 pthread_exit() 函數(shù)的含義不同,return 的含義是返回,它不僅可以用于線程執(zhí)行的函數(shù),普通函數(shù)也可以使用;pthread_exit() 函數(shù)的含義是線程退出,它專門用于結束某個線程的執(zhí)行。
在主線程(main() 函數(shù))中,return 和 pthread_exit() 函數(shù)的區(qū)別最明顯。舉個例子:
#include <stdio.h> #include <pthread.h> void *ThreadFun(void *arg) { sleep(5);//等待一段時間 printf("http://jb51.net\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í)行此程序,輸出結果為: 腳本之家
通過執(zhí)行結果可以看到,主線程正常執(zhí)行結束,myThread 線程并沒有輸出指定的數(shù)據(jù)。原因很簡單,主線程執(zhí)行速度很快,主線程最后執(zhí)行的 return 語句不僅會終止主線程執(zhí)行,還會終止其它子線程執(zhí)行。也就是說,myThread 線程還沒有執(zhí)行輸出語句就被終止了。
將上面程序中,main() 函數(shù)中的return 0;用如下語句替換:
pthread_exit(NULL);
重新編譯、執(zhí)行程序,運行結果為:
腳本之家
http://jb51.net
對比上面兩個執(zhí)行結果,我們可以得出的結論是:pthread_exit() 函數(shù)只會終止當前線程,不會影響進程中其它線程的執(zhí)行。
此外,pthread_exit() 可以自動調用線程清理程序(本質是一個由 pthread_cleanup_push() 指定的自定義函數(shù)),return 則不具備這個能力??傊趯嶋H場景中,如果想終止某個子線程執(zhí)行,強烈建議大家使用 pthread_exit() 函數(shù)。終止主線程時,return 和 pthread_exit() 函數(shù)發(fā)揮的功能不同,可以根據(jù)需要自行選擇。
到此這篇關于C語言中pthread_exit()函數(shù)實現(xiàn)終止線程的文章就介紹到這了,更多相關pthread_exit()終止線程內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的
相關文章
C++中類的成員函數(shù)及內聯(lián)函數(shù)使用及說明
這篇文章主要介紹了C++中類的成員函數(shù)及內聯(lián)函數(shù)使用及說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-11-11C++基于蔡基姆拉爾森計算公式實現(xiàn)由年月日確定周幾的方法示例
這篇文章主要介紹了C++基于蔡基姆拉爾森計算公式實現(xiàn)由年月日確定周幾的方法,涉及C++針對日期時間的數(shù)值運算相關操作技巧,需要的朋友可以參考下2017-07-07