關(guān)于多線程創(chuàng)建時(shí)CreateThread和_beginthreadex的區(qū)別
在 Win32 API 中,創(chuàng)建線程的基本函數(shù)是 CreateThread,而 _beginthread(ex) 是C++ 運(yùn)行庫的函數(shù)。為什么要有兩個(gè)呢?因?yàn)镃++ 運(yùn)行庫里面有一些函數(shù)使用了全局量,如果使用 CreateThread 的情況下使用這些C++ 運(yùn)行庫的函數(shù),就會(huì)出現(xiàn)不安全的問題。而 _beginthreadex 為這些全局變量做了處理,使得每個(gè)線程都有一份獨(dú)立的“全局”量。
所以,如果你的編程只調(diào)用 Win32 API/SDK ,就放心用 CreateThread;如果要用到C++ 運(yùn)行時(shí)間庫,那么就要使用 _beginthreadex ,并且需要在編譯環(huán)境中選擇 Use?MultiThread Lib/DLL。
C++ 運(yùn)行期庫有兩個(gè)創(chuàng)建線程的函數(shù),另一個(gè)是 _beginthread, 它們兩者的區(qū)別請(qǐng)自己去看MSDN:
通常他們的解釋都是這容易造成內(nèi)存泄漏。這個(gè)解釋本身是沒有錯(cuò)的,但是解釋得不夠完全和詳 細(xì)。以至于造成很多新手盲目的信任了那句話,在那里都是用_beginthreadex函數(shù),或者是裝作沒有看到使用CreateThread函數(shù)。曾經(jīng) 有一段時(shí)間我也對(duì)這個(gè)問題很是困惑,不知道到底用那個(gè)才是對(duì)的。因?yàn)槲也恢挂淮卧诤芏鄼?quán)威性的代碼中看到對(duì)CreateThread函數(shù)的直接調(diào)用。難道 是權(quán)威錯(cuò)了?? 抱著懷疑的態(tài)度查找了大量的資料和書籍,終于搞明白了這個(gè)問題的關(guān)鍵所在,在此做個(gè)說明,算是對(duì)那句話的一個(gè)完善。
?
關(guān)于_beginthreadex和CreateThread的區(qū)別我就不做說明了,這個(gè)很 容易找到的。我們只要知道一個(gè)問題:_beginthreadex是一個(gè)C運(yùn)行時(shí)庫的函數(shù),CreateThread是一個(gè)系統(tǒng)API函 數(shù),_beginthreadex內(nèi)部調(diào)用了CreateThread。只所以所有的書都強(qiáng)調(diào)內(nèi)存泄漏的問題是因?yàn)開beginthreadex函數(shù)在創(chuàng) 建線程的時(shí)候分配了一個(gè)堆結(jié)構(gòu)并和線程本身關(guān)聯(lián)起來,我們把這個(gè)結(jié)構(gòu)叫做tiddata結(jié)構(gòu),是通過線程本地存儲(chǔ)器TLS于線程本身關(guān)聯(lián)起來。我們傳入的 線程入口函數(shù)就保存在這個(gè)結(jié)構(gòu)中。tiddata的作用除了保存線程函數(shù)入口地址之外,還有一個(gè)重要的作用就是:C運(yùn)行時(shí)庫中有些函數(shù)需要通過這個(gè)結(jié)構(gòu)來 保存和獲取一些數(shù)據(jù),比如說errno之類的線程全局變量。這點(diǎn)才是最重要的。
當(dāng)一個(gè)線程調(diào)用一個(gè)要求tiddata結(jié)構(gòu)的運(yùn)行時(shí)庫函數(shù)的時(shí)候,將發(fā)生下面的情況:
運(yùn)行時(shí)庫函數(shù)試圖TlsGetv alue獲取線程數(shù)據(jù)塊的地址,如果沒有獲取到,函數(shù)就會(huì) 現(xiàn)場分配一個(gè) tiddata結(jié)構(gòu),并且和線程相關(guān)聯(lián),于是問題出現(xiàn)了,如果不通過_endthreadex函數(shù)來終結(jié)線程的話,這個(gè)結(jié)構(gòu)將不會(huì)被撤銷,內(nèi)存泄漏就會(huì)出 現(xiàn)了。但通常情況下,我們都不推薦使用_endthreadex函數(shù)來結(jié)束線程,因?yàn)槔锩姘薊xitThread調(diào)用。
?
找到了內(nèi)存泄漏的具體原因,我們可以這樣說:只要在創(chuàng)建的線程里面不使用一些要求tiddata結(jié)構(gòu)的運(yùn)行時(shí)庫函數(shù),我們的內(nèi)存時(shí)安全的。所以,前面說的那句話應(yīng)該這樣說才完善:
“絕對(duì)不要調(diào)用系統(tǒng)自帶的CreateThread函數(shù)創(chuàng)建新的線程,而應(yīng)該使用_beginthreadex,除非你在線程中絕不使用需要tiddata結(jié)構(gòu)的運(yùn)行時(shí)庫函數(shù)”
?
這個(gè)需要tiddata結(jié)構(gòu)的函數(shù)有點(diǎn)麻煩了,在侯捷的《win32多線程程序設(shè)計(jì)》一書中這樣說到:
”如果在除主線程之外的任何線程中進(jìn)行一下操作,你就應(yīng)該使用多線程版本的C runtime library,并使用_beginthreadex和_endthreadex:
1 使用malloc()和free(),或是new和delete
2 使用stdio.h或io.h里面聲明的任何函數(shù)
3 使用浮點(diǎn)變量或浮點(diǎn)運(yùn)算函數(shù)
4 調(diào)用任何一個(gè)使用了靜態(tài)緩沖區(qū)的runtime函數(shù),比如:asctime(),strtok()或rand()。