動態(tài)鏈接庫、靜態(tài)庫、import庫區(qū)別
動態(tài)鏈接庫(Dynamic Linked Library):
Windows為應用程序提供了豐富的函數(shù)調(diào)用,這些函數(shù)調(diào)用都包含在動態(tài)鏈接庫中。其中有3個最重要的DLL,Kernel32.dll,它包含用于管理內(nèi)存、進程和線程的各個函數(shù);
User32.dll,它包含用于執(zhí)行用戶界面任務(如窗口的創(chuàng)建和消息的傳送)的各個函數(shù);GDI32.dll,它包含用于畫圖和顯示文本的各個函數(shù)。
靜態(tài)庫(Static Library):
函數(shù)和數(shù)據(jù)被編譯進一個二進制文件(通常擴展名為.LIB)。在使用靜態(tài)庫的情況下,在編譯鏈接可執(zhí)行文件時,鏈接器從庫中復制這些函數(shù)和數(shù)據(jù)并把它們和應用程序的其它模塊
組合起來創(chuàng)建最終的可執(zhí)行文件(.EXE文件)。
導入庫(Import Library):
在使用動態(tài)鏈接庫的時候,往往提供兩個文件:一個引入庫和一個DLL。引入庫包含被DLL導出的函數(shù)和變量的符號名,DLL包含實際的函數(shù)和數(shù)據(jù)。在編譯鏈接可執(zhí)行文件時,只需
要鏈接引入庫,DLL中的函數(shù)代碼和數(shù)據(jù)并不復制到可執(zhí)行文件中,在運行的時候,再去加載DLL,訪問DLL中導出的函數(shù)。
在運行Windows程序時,它通過一個被稱作“動態(tài)鏈接”的進程與Windows相接。一個Windows的.EXE文件擁有它使用不同動態(tài)鏈接庫的引用,所使用的函數(shù)即在那里。當Windows程
序被加載到內(nèi)存中時,程序中的調(diào)用被指向DLL函數(shù)的入口,如果DLL不在內(nèi)存中,系統(tǒng)就將其加載到內(nèi)存中。
當鏈接Windows程序以產(chǎn)生一個可執(zhí)行文件時,你必須鏈接由編程環(huán)境提供的專門的“導入庫(import library)庫”。這些導入庫包含了動態(tài)鏈接庫名稱和所有Windows函數(shù)調(diào)用的
引用信息。鏈接程序使用該信息在.EXE文件中構(gòu)造一個表,當加載程序時,Windows使用它將調(diào)用轉(zhuǎn)換為Windows函數(shù)。
靜態(tài)庫與導入庫的區(qū)別:
導入庫和靜態(tài)庫的區(qū)別很大,他們實質(zhì)是不一樣的東西。靜態(tài)庫本身就包含了實際執(zhí)行代碼、符號表等等,而對于導入庫而言,其實際的執(zhí)行代碼位于動態(tài)庫中,導入庫只包含了
地址符號表等,確保程序找到對應函數(shù)的一些基本地址信息。
靜態(tài)鏈接與動態(tài)鏈接:
靜態(tài)鏈接方法:#pragma comment(lib, "test.lib") ,靜態(tài)鏈接的時候,載入代碼就會把程序會用到的動態(tài)代碼或動態(tài)代碼的地址確定下來
靜態(tài)庫的鏈接可以使用靜態(tài)鏈接,動態(tài)鏈接庫也可以使用這種方法鏈接導入庫
動態(tài)鏈接方法:LoadLibrary()/GetProcessAddress()和FreeLibrary(),使用這種方式的程序并不在一開始就完成動態(tài)鏈接,而是直到真正調(diào)用動態(tài)庫代碼時,載入程序才計算(被調(diào)用的那部分)動態(tài)代碼的邏輯地址,然后等到某個時候,程序又需要調(diào)用另外某塊動態(tài)代碼時,載入程序又去計算這部分代碼的邏輯地址,所以,這種方式使程序初始化時間較短,但運行期間的性能比不上靜態(tài)鏈接的程序。
在軟件開發(fā)的過程中,大家經(jīng)常會或多或少的使用別人編寫的或者系統(tǒng)提供的動態(tài)庫或靜態(tài)庫,但是究竟是使用靜態(tài)庫還是動態(tài)庫呢?他們的適用條件是什么呢?
簡單的說,靜態(tài)庫和應用程序編譯在一起,在任何情況下都能運行,而動態(tài)庫是動態(tài)鏈接,顧名思義就是在應用程序啟動的時候才會鏈接,所以,當用戶的系統(tǒng)上沒有該動態(tài)庫時,應用程序就會運行失敗。再看它們的特點:
動態(tài)庫:
1.共享:多個應用程序可以使用同一個動態(tài)庫,啟動多個應用程序的時候,只需要將動態(tài)庫加載到內(nèi)存一次即可;
2.開發(fā)模塊好:要求設計者對功能劃分的比較好。
靜態(tài)庫:代碼的裝載速度快,執(zhí)行速度也比較快,因為編譯時它只會把你需要的那部分鏈接進去,應用程序相對比較大。但是如果多個應用程序使用的話,會被裝載多次,浪費內(nèi)存。
總上,我個人認為,如果你的系統(tǒng)上有多個應用程序都使用該庫的話,就把它編譯成動態(tài)庫,這樣雖然剛啟動的時候加載比較慢,但是多任務的時候會比較節(jié)省內(nèi)存;如果你的系統(tǒng)上只有一到兩個應用使用該庫,并且使用的API比較少的話,就編譯成靜態(tài)庫吧,一般的靜態(tài)庫還可以進行裁剪編譯,這樣應用程序可能會比較大,但是啟動的速度會大大提高。