CreateThread()與beginthread()的區(qū)別詳細(xì)解析
我們知道在Windows下創(chuàng)建一個(gè)線程的方法有兩種,一種就是調(diào)用Windows API CreateThread()來(lái)創(chuàng)建線程;另外一種就是調(diào)用MSVC CRT的函數(shù)_beginthread()或_beginthreadex()來(lái)創(chuàng)建線程。相應(yīng)的退出線程也有兩個(gè)函數(shù)Windows API的ExitThread()和CRT的_endthread()。這兩套函數(shù)都是用來(lái)創(chuàng)建和退出線程的,它們有什么區(qū)別呢?
很多開(kāi)發(fā)者不清楚這兩者之間的關(guān)系,他們隨意選一個(gè)函數(shù)來(lái)用,發(fā)現(xiàn)也沒(méi)有什么大問(wèn)題,于是就忙于解決更為緊迫的任務(wù)去了,而沒(méi)有對(duì)它們進(jìn)行深究。等到有一天忽然發(fā)現(xiàn)一個(gè)程序運(yùn)行時(shí)間很長(zhǎng)的時(shí)候會(huì)有細(xì)微的內(nèi)存泄露,開(kāi)發(fā)者絕對(duì)不會(huì)想到是因?yàn)檫@兩套函數(shù)用混的結(jié)果。
根據(jù)Windows API和MSVC CRT的關(guān)系,可以看出來(lái)_beginthread()是對(duì)CreateThread()的包裝,它最終還是調(diào)用CreateThread()來(lái)創(chuàng)建線程。那么在_beginthread()調(diào)用CreateThread()之前做了什么呢?我們可以看一下_beginthread()的源代碼,它位于CRT源代碼中的thread.c。我們可以發(fā)現(xiàn)它在調(diào)用CreateThread()之前申請(qǐng)了一個(gè)叫_tiddata的結(jié)構(gòu),然后將這個(gè)結(jié)構(gòu)用_initptd()函數(shù)初始化之后傳遞給_beginthread()自己的線程入口函數(shù)_threadstart。_threadstart首先把由_beginthread()傳過(guò)來(lái)的_tiddata結(jié)構(gòu)指針保存到線程的顯式TLS數(shù)組,然后它調(diào)用用戶的線程入口真正開(kāi)始線程。在用戶線程結(jié)束之后,_threadstart()函數(shù)調(diào)用_endthread()結(jié)束線程。并且_threadstart還用__try/__except將用戶線程入口函數(shù)包起來(lái),用于捕獲所有未處理的信號(hào),并且將這些信號(hào)交給CRT處理。
所以除了信號(hào)之外,很明顯CRT包裝Windows API線程接口的最主要目的就是那個(gè)_tiddata。這個(gè)線程私有的結(jié)構(gòu)里面保存的是什么呢?我們可以從mtdll.h中找到它的定義,它里面保存的是諸如線程ID、線程句柄、erron、strtok()的前一次調(diào)用位置、rand()函數(shù)的種子、異常處理等與CRT有關(guān)的而且是線程私有的信息??梢?jiàn)MSVC CRT并沒(méi)有使用我們前面所說(shuō)的__declspec(thread)這種方式來(lái)定義線程私有變量,從而防止庫(kù)函數(shù)在多線程下失效,而是采用在堆上申請(qǐng)一個(gè)_tiddata結(jié)構(gòu),把線程私有變量放在結(jié)構(gòu)內(nèi)部,由顯式TLS保存_tiddata的指針。
了解了這些信息以后,我們應(yīng)該會(huì)想到一個(gè)問(wèn)題,那就是如果我們用CreateThread()創(chuàng)建一個(gè)線程然后調(diào)用CRT的strtok()函數(shù),按理說(shuō)應(yīng)該會(huì)出錯(cuò),因?yàn)閟trtok()所需要的_tiddata并不存在,可是我們好像從來(lái)沒(méi)碰到過(guò)這樣的問(wèn)題。查看strtok()函數(shù)就會(huì)發(fā)現(xiàn),當(dāng)一開(kāi)始調(diào)用_getptd()去得到線程的_tiddata結(jié)構(gòu)時(shí),這個(gè)函數(shù)如果發(fā)現(xiàn)線程沒(méi)有申請(qǐng)_tiddata結(jié)構(gòu),它就會(huì)申請(qǐng)這個(gè)結(jié)構(gòu)并且負(fù)責(zé)初始化。于是無(wú)論我們調(diào)用哪個(gè)函數(shù)創(chuàng)建線程,都可以安全調(diào)用所有需要_tiddata的函數(shù),因?yàn)橐坏┻@個(gè)結(jié)構(gòu)不存在,它就會(huì)被創(chuàng)建出來(lái)。
那么_tiddata在什么時(shí)候會(huì)被釋放呢?ExitThread()肯定不會(huì),因?yàn)樗静恢烙衉tiddata這樣一個(gè)結(jié)構(gòu)存在,那么很明顯是_endthread()釋放的,這也正是CRT的做法。不過(guò)我們很多時(shí)候會(huì)發(fā)現(xiàn),即使使用CreateThread()和ExitThread() (不調(diào)用ExitThread()直接退出線程函數(shù)的效果相同),也不會(huì)發(fā)現(xiàn)任何內(nèi)存泄露,這又是為什么呢?經(jīng)過(guò)仔細(xì)檢查之后,我們發(fā)現(xiàn)原來(lái)密碼在CRT DLL的入口函數(shù)DllMain中。我們知道,當(dāng)一個(gè)進(jìn)程/線程開(kāi)始或退出的時(shí)候,每個(gè)DLL的DllMain都會(huì)被調(diào)用一次,于是動(dòng)態(tài)鏈接版的CRT就有機(jī)會(huì)在DllMain中釋放線程的_tiddata??墒荄llMain只有當(dāng)CRT是動(dòng)態(tài)鏈接版的時(shí)候才起作用,靜態(tài)鏈接CRT是沒(méi)有DllMain的!這就是造成使用CreateThread()會(huì)導(dǎo)致內(nèi)存泄露的一種情況,在這種情況下,_tiddata在線程結(jié)束時(shí)無(wú)法釋放,造成了泄露。
我們可以用下面這個(gè)小程序來(lái)測(cè)試:
#include <Windows.h>
#include <process.h>
void thread(void *a)
{
char* r = strtok( "aaa", "b" );
ExitThread(0); // 這個(gè)函數(shù)是否調(diào)用都無(wú)所謂
}
int main(int argc, char* argv[])
{
while(1) {
CreateThread( 0, 0, (LPTHREAD_START_ROUTINE)thread, 0, 0, 0 );
Sleep( 5 );
}
return 0;
}
如果用動(dòng)態(tài)鏈接的CRT (/MD,/MDd)就不會(huì)有問(wèn)題,但是,如果使用靜態(tài)鏈接CRT (/MT,/MTd),運(yùn)行程序后在進(jìn)程管理器中觀察它就會(huì)發(fā)現(xiàn)內(nèi)存用量不停地上升,但是如果我們把thread()函數(shù)中的ExitThread()改成_endthread()就不會(huì)有問(wèn)題,因?yàn)開(kāi)endthread()會(huì)將_tiddata()釋放。
這個(gè)問(wèn)題可以總結(jié)為:當(dāng)使用CRT時(shí)(基本上所有的程序都使用CRT),請(qǐng)盡量使用_beginthread()/_beginthreadex()/_endthread()/_endthreadex()這組函數(shù)來(lái)創(chuàng)建線程。在MFC中,還有一組類似的函數(shù)是AfxBeginThread()和AfxEndThread(),根據(jù)上面的原理類推,它是MFC層面的線程包裝函數(shù),它們會(huì)維護(hù)線程與MFC相關(guān)的結(jié)構(gòu),當(dāng)我們使用MFC類庫(kù)時(shí),盡量使用它提供的線程包裝函數(shù)以保證程序運(yùn)行正確。
欄 目:C語(yǔ)言
本文標(biāo)題:CreateThread()與beginthread()的區(qū)別詳細(xì)解析
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/4121.html
您可能感興趣的文章
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10深入理解atoi()與itoa()函數(shù)的用法
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解
- 01-10基于atoi()與itoa()函數(shù)的內(nèi)部實(shí)現(xiàn)方法詳解
- 01-10Linux線程管理必備:解析互斥量與條件變量的詳解
- 01-10深入理解大數(shù)與高精度數(shù)的處理問(wèn)題
- 01-10深入理解數(shù)組指針與指針數(shù)組的區(qū)別
- 01-10C語(yǔ)言游戲必備:光標(biāo)定位與顏色設(shè)置的實(shí)現(xiàn)方法
- 01-10深入探討C語(yǔ)言中局部變量與全局變量在內(nèi)存中的存放位置
- 01-10linux c 查找使用庫(kù)的cflags與libs的方法詳解


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10C#中split用法實(shí)例總結(jié)
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10delphi制作wav文件的方法
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 04-02jquery與jsp,用jquery