如果在VC中創(chuàng)建一個(gè)控制臺(tái)的EXE和一個(gè)Win32的DLL,從DLL中導(dǎo)出一個(gè)函數(shù),該函數(shù)用new分配一塊內(nèi)存,返回其指針,然后在EXE中調(diào)用該函數(shù),獲得返回的指針,用delete釋放這塊內(nèi)存,就會(huì)引發(fā)斷言錯(cuò)誤。
產(chǎn)生這個(gè)問(wèn)題的原因是:EXE和DLL中分別靜態(tài)鏈接了C運(yùn)行時(shí)庫(kù),從而new和delete運(yùn)算符來(lái)自C運(yùn)行時(shí)庫(kù)的不同版本。C運(yùn)行時(shí)庫(kù)在管理堆內(nèi)存時(shí),會(huì)使用一些全局變量來(lái)跟蹤內(nèi)存分配情況,因此程序中鏈接的C運(yùn)行時(shí)庫(kù)必須唯一,否則就會(huì)引起不一致。
解決的辦法很簡(jiǎn)單:在EXE和DLL中都動(dòng)態(tài)鏈接C運(yùn)行時(shí)庫(kù),也就是在工程設(shè)置的Link面板選擇"忽略所有默認(rèn)的庫(kù)",再加入msvcrt.lib。
對(duì)這個(gè)問(wèn)題有兩種錯(cuò)誤的觀點(diǎn)需要澄清:一種以為EXE和DLL有不同的堆,實(shí)際上DLL總是被映射到加載它的進(jìn)程的地址空間,它沒(méi)有自己的堆;一種以為DLL和EXE相對(duì)于不同的起始地址,動(dòng)態(tài)鏈接的地址映射機(jī)制引起了前面的問(wèn)題,實(shí)際上DLL是和OBJ一樣的目標(biāo)模塊,每個(gè)目標(biāo)模塊都有自己的起始地址,但是鏈接成加載模塊以后就會(huì)統(tǒng)一到一個(gè)起始地址,一個(gè)目標(biāo)模塊對(duì)其它模塊的引用在鏈接前是以符號(hào)方式表示的,鏈接后會(huì)被修改成地址方式。靜態(tài)鏈接和動(dòng)態(tài)鏈接都會(huì)保證:加載模塊是統(tǒng)一編址的。
參考資料:
1. http://topic.csdn.net/t/20020714/19/873683.html
2. MSDN July 2000/Knowledge Base/Windows Development/Win32 Software Development Kit/HOWTO: Use the C Run-Time
3. 《操作系統(tǒng)-內(nèi)核與設(shè)計(jì)原理(第四版)》/William Stallings 著,魏迎梅等譯 電子工業(yè)出版社