以文本方式查看主題 - 曙海教育集團論壇 (http://www.hufushizhe.com/bbs/index.asp) -- VB語言 (http://www.hufushizhe.com/bbs/list.asp?boardid=77) ---- 技術(shù)討論總結(jié) -之- DLL和COM (http://www.hufushizhe.com/bbs/dispbbs.asp?boardid=77&id=2609) |
-- 作者:wangxinxin -- 發(fā)布時間:2010-12-14 14:52:04 -- 技術(shù)討論總結(jié) -之- DLL和COM 本周技術(shù)討論會主要議題是DLL和COM DLL是微軟創(chuàng)造的二進制級別代碼重用和應用程序管理的技術(shù). DLL有如下這些好處: 1. 由于DLL動態(tài)加載的特性, 使得DLL模塊和應用程序主干相剝離, 便于程序的升級 -----升級某個模塊的功能只需覆蓋某個DLL就可以了, 無需重新編譯全部代碼. 2. DLL的另一個好處也來自動態(tài)加載, 因此依賴于這個模塊的其他各類應用程序不需與DLL編譯在一起, 因此省出了更多的硬盤空間. 假如使用靜態(tài)鏈接, 則每一個應用程序都需包含相應的模塊鏡像, 對磁盤的浪費是很嚴重的. 3. 更為重要的, 使用DLL還有助于節(jié)省內(nèi)存. 因為系統(tǒng)為DLL提供了引用計數(shù), 所有加載DLL的進程共享DLL的頁面(當然數(shù)據(jù)不能共享,數(shù)據(jù)由進程自己保存.能夠共享的是相當于代碼段的頁面),因此DLL使得內(nèi)存的使用更有效. 4. 使用DLL可以實現(xiàn)多語言編程.例如VB的程序可以調(diào)用C++代碼實現(xiàn)的DLL模塊里面的函數(shù). DLL有兩種加載方式, 一種是隱示加載, 就是在應用程序里包含DLL編寫者提供的.h文件和編譯時產(chǎn)生的.lib文件. 這樣DLL將會在程序啟動時加載入進程. 可以想見, 如word這類的程序在啟動之時如此緩慢, 就是在不斷的加載各種DLL模塊. 另一種加載方式就是在程序中調(diào)用LoadLibrary函數(shù)顯示加載DLL, 然后通過調(diào)用GetProcAddress函數(shù)獲取相應函數(shù)的地址(即函數(shù)指針), 從而調(diào)用DLL內(nèi)的函數(shù). 這種做法有一個很好的優(yōu)點就是不必所有的DLL都在啟動時加載, 很多DLL模塊都可以在需要使用的時候再臨時加載, 有些DLL假如沒有用到可以一直不加載. 我個人認為這是很好的用戶體驗. 我不理解的是為什么MS的各類產(chǎn)品沒有使用這種方式進行程序設計.或許他已經(jīng)用了, 但是啟動時候必須加載的DLL項目太多?我就不再猜測了.. 顯示加載的DLL需要手動卸載, 調(diào)用FreeLibrary函數(shù)卸載它. 與普通的EXE一樣, DLL也有一個進入點函數(shù), 名字叫做DllMain, 這個函數(shù)接受三個參數(shù), HINSTANCE hInst, DWORD fdwReason, PVOID fImpLoad 我在這里只討論第二個參數(shù)fdwReason. 它可以是4個值: DLL_PROCESS_ATTACH, DLL_PROCESS_DETACH, DLL_THREAD_ATTACH, DLL_THREAD_DETACH 分別表示DLL首次加載,DLL從進程中釋放, DLL線程的構(gòu)造和終止. 其實系統(tǒng)為DLL提供了引用計數(shù), DLL只加載入RAM一次, 所有進程共享DLL的各個頁面, 所以只有第一次加載會調(diào)用帶有DLL_PROCESS_ATTACH的DllMain. 但每次DLL從進程中釋放都會調(diào)用帶DLL_PROCESS_DETACH的DllMain, 這是因為進程保存了DLL里面的一些變量可能需要釋放掉. 因此DllMain可以阻止進程結(jié)束.它可以在DLL_PROCESS_DETACH消息中放一個死循環(huán),等待所有資源都被釋放掉才退出. 如果有些資源出了問題, 就可能導致DLL無法正常退出.這就是為什么有些程序即使你去點小紅叉叉也結(jié)束不了進程的原因.這種情況可以調(diào)用TerminateProcess函數(shù)強關(guān)進程.這就是為什么通過任務管理器仍然能結(jié)束進程的原因(但是這樣做有可能丟失數(shù)據(jù)...) |