成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

開發(fā) 后端
這次分享的宗旨是——讓大家學(xué)會(huì)創(chuàng)建與使用靜態(tài)庫、動(dòng)態(tài)庫,知道靜態(tài)庫與動(dòng)態(tài)庫的區(qū)別,知道使用的時(shí)候如何選擇。這里不深入介紹靜態(tài)庫、動(dòng)態(tài)庫的底層格式,內(nèi)存布局等,有興趣的同學(xué),推薦一本書《程序員的自我修養(yǎng)——鏈接、裝載與庫》。

這次分享的宗旨是——讓大家學(xué)會(huì)創(chuàng)建與使用靜態(tài)庫、動(dòng)態(tài)庫,知道靜態(tài)庫與動(dòng)態(tài)庫的區(qū)別,知道使用的時(shí)候如何選擇。這里不深入介紹靜態(tài)庫、動(dòng)態(tài)庫的底層格式,內(nèi)存布局等,有興趣的同學(xué),推薦一本書《程序員的自我修養(yǎng)——鏈接、裝載與庫》。

什么是庫

庫是寫好的現(xiàn)有的,成熟的,可以復(fù)用的代碼。現(xiàn)實(shí)中每個(gè)程序都要依賴很多基礎(chǔ)的底層庫,不可能每個(gè)人的代碼都從零開始,因此庫的存在意義非同尋常。

本質(zhì)上來說庫是一種可執(zhí)行代碼的二進(jìn)制形式,可以被操作系統(tǒng)載入內(nèi)存執(zhí)行。庫有兩種:靜態(tài)庫(.a、.lib)和動(dòng)態(tài)庫(.so、.dll)。

所謂靜態(tài)、動(dòng)態(tài)是指鏈接。回顧一下,將一個(gè)程序編譯成可執(zhí)行程序的步驟:

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

靜態(tài)庫

之所以成為【靜態(tài)庫】,是因?yàn)樵阪溄与A段,會(huì)將匯編生成的目標(biāo)文件.o與引用到的庫一起鏈接打包到可執(zhí)行文件中。因此對(duì)應(yīng)的鏈接方式稱為靜態(tài)鏈接。

試想一下,靜態(tài)庫與匯編生成的目標(biāo)文件一起鏈接為可執(zhí)行文件,那么靜態(tài)庫必定跟.o文件格式相似。其實(shí)一個(gè)靜態(tài)庫可以簡(jiǎn)單看成是一組目標(biāo)文件(.o/.obj文件)的集合,即很多目標(biāo)文件經(jīng)過壓縮打包后形成的一個(gè)文件。靜態(tài)庫特點(diǎn)總結(jié):

  • 靜態(tài)庫對(duì)函數(shù)庫的鏈接是放在編譯時(shí)期完成的。

  • 程序在運(yùn)行時(shí)與函數(shù)庫再無瓜葛,移植方便。

  • 浪費(fèi)空間和資源,因?yàn)樗邢嚓P(guān)的目標(biāo)文件與牽涉到的函數(shù)庫被鏈接合成一個(gè)可執(zhí)行文件。

下面編寫一些簡(jiǎn)單的四則運(yùn)算C++類,將其編譯成靜態(tài)庫給他人用,頭文件如下所示:

  1. #pragma once 
  2. class StaticMath 
  3. public
  4.     StaticMath(void); 
  5.     ~StaticMath(void); 
  6.  
  7.     static double add(double a, double b);//加法 
  8.     static double sub(double a, double b);//減法 
  9.     static double mul(double a, double b);//乘法 
  10.     static double div(double a, double b);//除法 
  11.  
  12.     void print(); 
  13. }; 

inux下使用ar工具、Windows下vs使用lib.exe,將目標(biāo)文件壓縮到一起,并且對(duì)其進(jìn)行編號(hào)和索引,以便于查找和檢索。一般創(chuàng)建靜態(tài)庫的步驟如圖所示:

#pragma once class StaticMath { public:     StaticMath(void);     ~StaticMath(void);      static double add(double a, double b);//加法     static double sub(double a, double b);//減法     static double mul(double a, double b);//乘法     static double div(double a, double b);//除法      void print(); };

#p#

Linux下創(chuàng)建與使用靜態(tài)庫

Linux靜態(tài)庫命名規(guī)則

Linux靜態(tài)庫命名規(guī)范,必須是”lib[your_library_name].a”:lib為前綴,中間是靜態(tài)庫名,擴(kuò)展名為.a。

創(chuàng)建靜態(tài)庫(.a)

通過上面的流程可以知道,Linux創(chuàng)建靜態(tài)庫過程如下:

  • 首先,將代碼文件編譯成目標(biāo)文件.o(StaticMath.o)

g++ -c StaticMath.cpp

注意帶參數(shù)-c,否則直接編譯為可執(zhí)行文件

  • 然后,通過ar工具將目標(biāo)文件打包成.a靜態(tài)庫文件

ar -crv libstaticmath.a StaticMath.o

生成靜態(tài)庫libstaticmath.a

Linux下創(chuàng)建與使用靜態(tài)庫

大一點(diǎn)的項(xiàng)目會(huì)編寫makefile文件(CMake等等工程管理工具)來生成靜態(tài)庫,輸入多個(gè)命令太麻煩了。

使用靜態(tài)庫

編寫使用上面創(chuàng)建的靜態(tài)庫的測(cè)試代碼:

  1. #include "StaticMath.h" 
  2. #include <iostream> 
  3. using namespace std; 
  4.  
  5. int main(int argc, char* argv[]) 
  6.     double a = 10; 
  7.     double b = 2; 
  8.  
  9.     cout << "a + b = " << StaticMath::add(a, b) << endl; 
  10.     cout << "a - b = " << StaticMath::sub(a, b) << endl; 
  11.     cout << "a * b = " << StaticMath::mul(a, b) << endl; 
  12.     cout << "a / b = " << StaticMath::div(a, b) << endl; 
  13.  
  14.     StaticMath sm; 
  15.     sm.print(); 
  16.  
  17.     system("pause"); 
  18.     return 0; 

Linux下使用靜態(tài)庫,只需要在編譯的時(shí)候,指定靜態(tài)庫的搜索路徑(-L選項(xiàng))、指定靜態(tài)庫名(不需要lib前綴和.a后綴,-l選項(xiàng))。

# g++ TestStaticLibrary.cpp -L../StaticLibrary -lstaticmath

  • -L:表示要連接的庫所在目錄

  • -l:指定鏈接時(shí)需要的動(dòng)態(tài)庫,編譯器查找動(dòng)態(tài)連接庫時(shí)有隱含的命名規(guī)則,即在給出的名字前面加上lib,后面加上.a或.so來確定庫的名稱。

#p#

Windows下創(chuàng)建與使用靜態(tài)庫

創(chuàng)建靜態(tài)庫(.lib

如果是使用VS命令行生成靜態(tài)庫,也是分兩個(gè)步驟來生成程序:

  • 首先,通過使用帶編譯器選項(xiàng) /c 的 Cl.exe 編譯代碼 (cl /c StaticMath.cpp),創(chuàng)建名為“StaticMath.obj”的目標(biāo)文件。

  • 然后,使用庫管理器 Lib.exe 鏈接代碼 (lib StaticMath.obj),創(chuàng)建靜態(tài)庫StaticMath.lib。

當(dāng)然,我們一般不這么用,使用VS工程設(shè)置更方便。創(chuàng)建win32控制臺(tái)程序時(shí),勾選靜態(tài)庫類型;打開工程“屬性面板”→”配置屬性”→”常規(guī)”,配置類型選擇靜態(tài)庫。

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

Build項(xiàng)目即可生成靜態(tài)庫。

使用靜態(tài)庫

測(cè)試代碼Linux下面的一樣。有3種使用方法:

方法一:

在VS中使用靜態(tài)庫方法:

  • 工程“屬性面板”→“通用屬性”→“框架和引用”→”添加引用”,將顯示“添加引用”對(duì)話框。 “項(xiàng)目”選項(xiàng)卡列出了當(dāng)前解決方案中的各個(gè)項(xiàng)目以及可以引用的所有庫。 在“項(xiàng)目”選項(xiàng)卡中,選擇 StaticLibrary。 單擊“確定”。

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

  • 添加StaticMath.h 頭文件目錄,必須修改包含目錄路徑。打開工程“屬性面板”→”配置屬性”→“C/C++”→” 常規(guī)”,在“附加包含目錄”屬性值中,鍵入StaticMath.h 頭文件所在目錄的路徑或?yàn)g覽至該目錄。

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

編譯運(yùn)行OK。

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

如果引用的靜態(tài)庫不是在同一解決方案下的子工程,而是使用第三方提供的靜態(tài)庫lib和頭文件,上面的方法設(shè)置不了。還有2中方法設(shè)置都可行。

方法二:

打開工程“屬性面板”→”配置屬性”→ “鏈接器”→ ”命令行”,輸入靜態(tài)庫的完整路徑即可。

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

方法三:

  • “屬性面板”→”配置屬性”→“鏈接器”→”常規(guī)”,附加依賴庫目錄中輸入,靜態(tài)庫所在目錄;

  • “屬性面板”→”配置屬性”→“鏈接器”→”輸入”,附加依賴庫中輸入靜態(tài)庫名StaticLibrary.lib。

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

#p#

動(dòng)態(tài)庫

通過上面的介紹發(fā)現(xiàn)靜態(tài)庫,容易使用和理解,也達(dá)到了代碼復(fù)用的目的,那為什么還需要?jiǎng)討B(tài)庫呢?

為什么還需要?jiǎng)討B(tài)庫?

為什么需要?jiǎng)討B(tài)庫,其實(shí)也是靜態(tài)庫的特點(diǎn)導(dǎo)致。

  • 空間浪費(fèi)是靜態(tài)庫的一個(gè)問題。

動(dòng)態(tài)庫

 

  • 另一個(gè)問題是靜態(tài)庫對(duì)程序的更新、部署和發(fā)布頁會(huì)帶來麻煩。如果靜態(tài)庫liba.lib更新了,所以使用它的應(yīng)用程序都需要重新編譯、發(fā)布給用戶(對(duì)于玩家來說,可能是一個(gè)很小的改動(dòng),卻導(dǎo)致整個(gè)程序重新下載,全量更新)。

動(dòng)態(tài)庫在程序編譯時(shí)并不會(huì)被連接到目標(biāo)代碼中,而是在程序運(yùn)行是才被載入。不同的應(yīng)用程序如果調(diào)用相同的庫,那么在內(nèi)存里只需要有一份該共享庫的實(shí) 例,規(guī)避了空間浪費(fèi)問題。動(dòng)態(tài)庫在程序運(yùn)行是才被載入,也解決了靜態(tài)庫對(duì)程序的更新、部署和發(fā)布頁會(huì)帶來麻煩。用戶只需要更新動(dòng)態(tài)庫即可,增量更新。 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

動(dòng)態(tài)庫特點(diǎn)總結(jié):

  • 動(dòng)態(tài)庫把對(duì)一些庫函數(shù)的鏈接載入推遲到程序運(yùn)行的時(shí)期。

  • 可以實(shí)現(xiàn)進(jìn)程之間的資源共享。(因此動(dòng)態(tài)庫也稱為共享庫)

  • 將一些程序升級(jí)變得簡(jiǎn)單。

  • 甚至可以真正做到鏈接載入完全由程序員在程序代碼中控制(顯示調(diào)用)。

Window與Linux執(zhí)行文件格式不同,在創(chuàng)建動(dòng)態(tài)庫的時(shí)候有一些差異。

  • 在Windows系統(tǒng)下的執(zhí)行文件格式是PE格式,動(dòng)態(tài)庫需要一個(gè)DllMain函數(shù)做出初始化的入口,通常在導(dǎo)出函數(shù)的聲明時(shí)需要有_declspec(dllexport)關(guān)鍵字。

  • Linux下gcc編譯的執(zhí)行文件默認(rèn)是ELF格式,不需要初始化入口,亦不需要函數(shù)做特別的聲明,編寫比較方便。

與創(chuàng)建靜態(tài)庫不同的是,不需要打包工具(ar、lib.exe),直接使用編譯器即可創(chuàng)建動(dòng)態(tài)庫。

#p#

Linux下創(chuàng)建與使用動(dòng)態(tài)庫

linux動(dòng)態(tài)庫的命名規(guī)則

動(dòng)態(tài)鏈接庫的名字形式為 libxxx.so,前綴是lib,后綴名為“.so”。

  • 針對(duì)于實(shí)際庫文件,每個(gè)共享庫都有個(gè)特殊的名字“soname”。在程序啟動(dòng)后,程序通過這個(gè)名字來告訴動(dòng)態(tài)加載器該載入哪個(gè)共享庫。

  • 在文件系統(tǒng)中,soname僅是一個(gè)鏈接到實(shí)際動(dòng)態(tài)庫的鏈接。對(duì)于動(dòng)態(tài)庫而言,每個(gè)庫實(shí)際上都有另一個(gè)名字給編譯器來用。它是一個(gè)指向?qū)嶋H庫鏡像文件的鏈接文件(lib+soname+.so)。

創(chuàng)建動(dòng)態(tài)庫(.so)

編寫四則運(yùn)算動(dòng)態(tài)庫代碼:

  1. #pragma once 
  2. class DynamicMath 
  3. public
  4.         DynamicMath(void); 
  5.         ~DynamicMath(void); 
  6.  
  7.         static double add(double a, double b); 
  8.         static double sub(double a, double b); 
  9.         static double mul(double a, double b); 
  10.         static double div(double a, double b); 
  11.         void print(); 
  12. }; 

首先,生成目標(biāo)文件,此時(shí)要加編譯器選項(xiàng)-fpic

g++ -fPIC -c DynamicMath.cpp

-fPIC 創(chuàng)建與地址無關(guān)的編譯程序(pic,position independent code),是為了能夠在多個(gè)應(yīng)用程序間共享。

  • 然后,生成動(dòng)態(tài)庫,此時(shí)要加鏈接器選項(xiàng)-shared

g++ -shared -o libdynmath.so DynamicMath.o

-shared指定生成動(dòng)態(tài)鏈接庫。

其實(shí)上面兩個(gè)步驟可以合并為一個(gè)命令:

g++ -fPIC -shared -o libdynmath.so DynamicMath.cpp

使用動(dòng)態(tài)庫

編寫使用動(dòng)態(tài)庫的測(cè)試代碼:

  1. #include "../DynamicLibrary/DynamicMath.h" 
  2.  
  3. #include <iostream> 
  4. using namespace std; 
  5.  
  6. int main(int argc, char* argv[]) 
  7.     double a = 10; 
  8.     double b = 2; 
  9.  
  10.     cout << "a + b = " << DynamicMath::add(a, b) << endl; 
  11.     cout << "a - b = " << DynamicMath::sub(a, b) << endl; 
  12.     cout << "a * b = " << DynamicMath::mul(a, b) << endl; 
  13.     cout << "a / b = " << DynamicMath::div(a, b) << endl; 
  14.  
  15.     DynamicMath dyn; 
  16.     dyn.print(); 
  17.     return 0; 

引用動(dòng)態(tài)庫編譯成可執(zhí)行文件(跟靜態(tài)庫方式一樣):

g++ TestDynamicLibrary.cpp -L../DynamicLibrary -ldynmath

然后運(yùn)行:./a.out,發(fā)現(xiàn)竟然報(bào)錯(cuò)了!!!

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

可能大家會(huì)猜測(cè),是因?yàn)閯?dòng)態(tài)庫跟測(cè)試程序不是一個(gè)目錄,那我們驗(yàn)證下是否如此:

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

發(fā)現(xiàn)還是報(bào)錯(cuò)!!!那么,在執(zhí)行的時(shí)候是如何定位共享庫文件的呢?

1) 當(dāng)系統(tǒng)加載可執(zhí)行代碼時(shí)候,能夠知道其所依賴的庫的名字,但是還需要知道絕對(duì)路徑。此時(shí)就需要系統(tǒng)動(dòng)態(tài)載入器(dynamic linker/loader)。

2) 對(duì)于elf格式的可執(zhí)行程序,是由ld-linux.so*來完成的,它先后搜索elf文件的 DT_RPATH段—環(huán)境變量LD_LIBRARY_PATH—/etc/ld.so.cache文件列表—/lib/,/usr/lib 目錄找到庫文件后將其載入內(nèi)存。

如何讓系統(tǒng)能夠找到它:

  • 如果安裝在/lib或者/usr/lib下,那么ld默認(rèn)能夠找到,無需其他操作。

  • 如果安裝在其他目錄,需要將其添加到/etc/ld.so.cache文件中,步驟如下:

  • 1. 編輯/etc/ld.so.conf文件,加入庫文件所在目錄的路徑

  • 2. 運(yùn)行l(wèi)dconfig ,該命令會(huì)重建/etc/ld.so.cache文件

我們將創(chuàng)建的動(dòng)態(tài)庫復(fù)制到/usr/lib下面,然后運(yùn)行測(cè)試程序。

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

#p#

Windows下創(chuàng)建與使用動(dòng)態(tài)庫

創(chuàng)建動(dòng)態(tài)庫(.dll)

與Linux相比,在Windows系統(tǒng)下創(chuàng)建動(dòng)態(tài)庫要稍微麻煩一些。首先,需要一個(gè)DllMain函數(shù)做出初始化的入口(創(chuàng)建win32控制臺(tái)程序時(shí),勾選DLL類型會(huì)自動(dòng)生成這個(gè)文件): 

  1. // dllmain.cpp : Defines the entry point for the DLL application. 
  2. #include "stdafx.h" 
  3.  
  4. BOOL APIENTRY DllMain( HMODULE hModule, 
  5.                        DWORD  ul_reason_for_call, 
  6.                        LPVOID lpReserved 
  7.                      ) 
  8.     switch (ul_reason_for_call) 
  9.     { 
  10.     case DLL_PROCESS_ATTACH: 
  11.     case DLL_THREAD_ATTACH: 
  12.     case DLL_THREAD_DETACH: 
  13.     case DLL_PROCESS_DETACH: 
  14.         break
  15.     } 
  16.     return TRUE; 

通常在導(dǎo)出函數(shù)的聲明時(shí)需要有_declspec(dllexport)關(guān)鍵字:

  1. #pragma once 
  2. class DynamicMath 
  3. public
  4.     __declspec(dllexport) DynamicMath(void); 
  5.     __declspec(dllexport) ~DynamicMath(void); 
  6.  
  7.     static __declspec(dllexportdouble add(double a, double b);//加法 
  8.     static __declspec(dllexportdouble sub(double a, double b);//減法 
  9.     static __declspec(dllexportdouble mul(double a, double b);//乘法 
  10.     static __declspec(dllexportdouble div(double a, double b);//除法 
  11.  
  12.     __declspec(dllexportvoid print(); 
  13. }; 

生成動(dòng)態(tài)庫需要設(shè)置工程屬性,打開工程“屬性面板”→”配置屬性”→”常規(guī)”,配置類型選擇動(dòng)態(tài)庫。

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

Build項(xiàng)目即可生成動(dòng)態(tài)庫。

使用動(dòng)態(tài)庫

創(chuàng)建win32控制臺(tái)測(cè)試程序:

  1. #include "stdafx.h" 
  2. #include "DynamicMath.h" 
  3.  
  4. #include <iostream> 
  5. using namespace std; 
  6.  
  7. int _tmain(int argc, _TCHAR* argv[]) 
  8.     double a = 10; 
  9.     double b = 2; 
  10.  
  11.     cout << "a + b = " << DynamicMath::add(a, b) << endl; 
  12.     cout << "a - b = " << DynamicMath::sub(a, b) << endl; 
  13.     cout << "a * b = " << DynamicMath::mul(a, b) << endl; 
  14.     cout << "a / b = " << DynamicMath::div(a, b) << endl; 
  15.  
  16.     DynamicMath dyn; 
  17.     dyn.print(); 
  18.  
  19.     system("pause"); 
  20.     return 0; 

方法一:

  • 工程“屬性面板”→“通用屬性”→“框架和引用”→”添加引用”,將顯示“添加引用”對(duì)話框。“項(xiàng)目”選項(xiàng)卡列出了當(dāng)前解決方案中的各個(gè)項(xiàng)目以及可以引用的所有庫。 在“項(xiàng)目”選項(xiàng)卡中,選擇 DynamicLibrary。 單擊“確定”。

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

  • 添加DynamicMath.h 頭文件目錄,必須修改包含目錄路徑。打開工程“屬性面板”→”配置屬性”→“C/C++”→” 常規(guī)”,在“附加包含目錄”屬性值中,鍵入DynamicMath.h 頭文件所在目錄的路徑或?yàn)g覽至該目錄。

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

編譯運(yùn)行OK。

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

方法二:

  • “屬性面板”→”配置屬性”→“鏈接器”→”常規(guī)”,附加依賴庫目錄中輸入,動(dòng)態(tài)庫所在目錄;

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

  • “屬性面板”→”配置屬性”→“鏈接器”→”輸入”,附加依賴庫中輸入動(dòng)態(tài)庫編譯出來的DynamicLibrary.lib。

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

這里可能大家有個(gè)疑問,動(dòng)態(tài)庫怎么還有一個(gè)DynamicLibrary.lib文件?即無論是靜態(tài)鏈接庫還是動(dòng)態(tài)鏈接庫,最后都有l(wèi)ib文件,那么兩者區(qū)別是什么呢?其實(shí),兩個(gè)是完全不一樣的東西。

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

StaticLibrary.lib的大小為190KB,DynamicLibrary.lib的大小為3KB,靜態(tài)庫對(duì)應(yīng)的lib文件叫靜態(tài)庫, 動(dòng)態(tài)庫對(duì)應(yīng)的lib文件叫【導(dǎo)入庫】。實(shí)際上靜態(tài)庫本身就包含了實(shí)際執(zhí)行代碼、符號(hào)表等等,而對(duì)于導(dǎo)入庫而言,其實(shí)際的執(zhí)行代碼位于動(dòng)態(tài)庫中,導(dǎo)入庫只包 含了地址符號(hào)表等,確保程序找到對(duì)應(yīng)函數(shù)的一些基本地址信息。

#p#

動(dòng)態(tài)庫的顯式調(diào)用

上面介紹的動(dòng)態(tài)庫使用方法和靜態(tài)庫類似屬于隱式調(diào)用,編譯的時(shí)候指定相應(yīng)的庫和查找路徑。其實(shí),動(dòng)態(tài)庫還可以顯式調(diào)用。【在C語言中】,顯示調(diào)用一個(gè)動(dòng)態(tài)庫輕而易舉!

在Linux下顯式調(diào)用動(dòng)態(tài)庫

#include <dlfcn.h>,提供了下面幾個(gè)接口:

  • void * dlopen( const char * pathname, int mode ):函數(shù)以指定模式打開指定的動(dòng)態(tài)連接庫文件,并返回一個(gè)句柄給調(diào)用進(jìn)程。

  • void* dlsym(void* handle,const char* symbol):dlsym根據(jù)動(dòng)態(tài)鏈接庫操作句柄(pHandle)與符號(hào)(symbol),返回符號(hào)對(duì)應(yīng)的地址。使用這個(gè)函數(shù)不但可以獲取函數(shù)地址,也可以獲取變量地址。

  • int dlclose (void *handle):dlclose用于關(guān)閉指定句柄的動(dòng)態(tài)鏈接庫,只有當(dāng)此動(dòng)態(tài)鏈接庫的使用計(jì)數(shù)為0時(shí),才會(huì)真正被系統(tǒng)卸載。

  • const char *dlerror(void):當(dāng)動(dòng)態(tài)鏈接庫操作函數(shù)執(zhí)行失敗時(shí),dlerror可以返回出錯(cuò)信息,返回值為NULL時(shí)表示操作函數(shù)執(zhí)行成功。

在Windows下顯式調(diào)用動(dòng)態(tài)庫

應(yīng)用程序必須進(jìn)行函數(shù)調(diào)用以在運(yùn)行時(shí)顯式加載 DLL。為顯式鏈接到 DLL,應(yīng)用程序必須:

  • 調(diào)用 LoadLibrary(或相似的函數(shù))以加載 DLL 和獲取模塊句柄。

  • 調(diào)用 GetProcAddress,以獲取指向應(yīng)用程序要調(diào)用的每個(gè)導(dǎo)出函數(shù)的函數(shù)指針。由于應(yīng)用程序是通過指針調(diào)用 DLL 的函數(shù),編譯器不生成外部引用,故無需與導(dǎo)入庫鏈接。

  • 使用完 DLL 后調(diào)用 FreeLibrary。

顯式調(diào)用C++動(dòng)態(tài)庫注意點(diǎn)

對(duì)C++來說,情況稍微復(fù)雜。顯式加載一個(gè)C++動(dòng)態(tài)庫的困難一部分是因?yàn)镃++的name mangling;另一部分是因?yàn)闆]有提供一個(gè)合適的API來裝載類,在C++中,您可能要用到庫中的一個(gè)類,而這需要?jiǎng)?chuàng)建該類的一個(gè)實(shí)例,這不容易做到。

name mangling可以通過extern “C”解決。C++有個(gè)特定的關(guān)鍵字用來聲明采用C binding的函數(shù):extern “C” 。用 extern “C”聲明的函數(shù)將使用函數(shù)名作符號(hào)名,就像C函數(shù)一樣。因此,只有非成員函數(shù)才能被聲明為extern “C”,并且不能被重載。盡管限制多多,extern “C”函數(shù)還是非常有用,因?yàn)樗鼈兛梢韵驝函數(shù)一樣被dlopen動(dòng)態(tài)加載。冠以extern “C”限定符后,并不意味著函數(shù)中無法使用C++代碼了,相反,它仍然是一個(gè)完全的C++函數(shù),可以使用任何C++特性和各種類型的參數(shù)。

另外如何從C++動(dòng)態(tài)庫中獲取類,附上幾篇相關(guān)文章,但我并不建議這么做:

  • 《LoadLibrary調(diào)用DLL中的Class》:http://www.cppblog.com/codejie/archive/2009/09/24/97141.html

  • 《C++ dlopen mini HOWTO》:http://blog.csdn.net/denny_233/article/details/7255673

“顯式”使用C++動(dòng)態(tài)庫中的Class是非常繁瑣和危險(xiǎn)的事情,因此能用“隱式”就不要用“顯式”,能靜態(tài)就不要用動(dòng)態(tài)。

附件:Linux下庫相關(guān)命令

g++(gcc)編譯選項(xiàng)

  • -shared :指定生成動(dòng)態(tài)鏈接庫。

  • -static :指定生成靜態(tài)鏈接庫。

  • -fPIC :表示編譯為位置獨(dú)立的代碼,用于編譯共享庫。目標(biāo)文件需要?jiǎng)?chuàng)建成位置無關(guān)碼, 念上就是在可執(zhí)行程序裝載它們的時(shí)候,它們可以放在可執(zhí)行程序的內(nèi)存里的任何地方。

  • -L. :表示要連接的庫所在的目錄。

  • -l:指定鏈接時(shí)需要的動(dòng)態(tài)庫。編譯器查找動(dòng)態(tài)連接庫時(shí)有隱含的命名規(guī)則,即在給出的名字前面加上lib,后面加上.a/.so來確定庫的名稱。

  • -Wall :生成所有警告信息。

  • -ggdb :此選項(xiàng)將盡可能的生成gdb 的可以使用的調(diào)試信息。

  • -g :編譯器在編譯的時(shí)候產(chǎn)生調(diào)試信息。

  • -c :只激活預(yù)處理、編譯和匯編,也就是把程序做成目標(biāo)文件(.o文件) 。

  • -Wl,options :把參數(shù)(options)傳遞給鏈接器ld 。如果options 中間有逗號(hào),就將options分成多個(gè)選項(xiàng),然后傳遞給鏈接程序。

nm命令

有時(shí)候可能需要查看一個(gè)庫中到底有哪些函數(shù),nm命令可以打印出庫中的涉及到的所有符號(hào)。庫既可以是靜態(tài)的也可以是動(dòng)態(tài)的。nm列出的符號(hào)有很多,常見的有三種:

  • 一種是在庫中被調(diào)用,但并沒有在庫中定義(表明需要其他庫支持),用U表示;

  • 一種是庫中定義的函數(shù),用T表示,這是最常見的;

  • 一種是所謂的弱態(tài)”符號(hào),它們雖然在庫中被定義,但是可能被其他庫中的同名符號(hào)覆蓋,用W表示。

$nm libhello.h

ldd命令

ldd命令可以查看一個(gè)可執(zhí)行程序依賴的共享庫,例如我們編寫的四則運(yùn)算動(dòng)態(tài)庫依賴下面這些庫:

 

C++靜態(tài)庫與動(dòng)態(tài)庫深入研究

總結(jié)

二者的不同點(diǎn)在于代碼被載入的時(shí)刻不同。

  • 靜態(tài)庫在程序編譯時(shí)會(huì)被連接到目標(biāo)代碼中,程序運(yùn)行時(shí)將不再需要該靜態(tài)庫,因此體積較大。

  • 動(dòng)態(tài)庫在程序編譯時(shí)并不會(huì)被連接到目標(biāo)代碼中,而是在程序運(yùn)行是才被載入,因此在程序運(yùn)行時(shí)還需要?jiǎng)討B(tài)庫存在,因此代碼體積較小。

動(dòng)態(tài)庫的好處是,不同的應(yīng)用程序如果調(diào)用相同的庫,那么在內(nèi)存里只需要有一份該共享庫的實(shí)例。帶來好處的同時(shí),也會(huì)有問題!如經(jīng)典的DLL Hell問題,關(guān)于如何規(guī)避動(dòng)態(tài)庫管理問題,可以自行查找相關(guān)資料。

責(zé)任編輯:王雪燕 來源: 吳秦(Tyler)
相關(guān)推薦

2011-12-15 10:43:20

JavaNIO

2010-08-26 15:48:21

DB2優(yōu)化數(shù)據(jù)庫

2016-01-12 18:04:55

Java異常研究

2017-06-06 11:29:23

Java異常研究與分析

2022-05-11 09:03:05

CSS容器文本換行

2011-06-01 10:58:54

Android Service

2010-11-18 17:24:27

Oracle旋轉(zhuǎn)ins

2023-03-15 15:58:11

Python動(dòng)態(tài)庫C++

2018-12-24 15:00:58

混合云多云云采用

2010-11-23 16:35:59

MySQL刪除多表數(shù)據(jù)

2017-06-09 14:30:31

Openstack N對(duì)象模型數(shù)據(jù)庫

2010-05-31 17:45:50

MySQL行鎖

2010-11-26 11:57:35

MySQL結(jié)果字符串

2010-06-21 13:07:14

2022-04-19 08:28:34

main函數(shù)

2009-10-20 10:17:50

綜合布線系統(tǒng)驗(yàn)收

2010-08-03 17:36:55

DB2數(shù)據(jù)庫

2011-06-09 11:11:35

QT 靜態(tài)庫 動(dòng)態(tài)庫

2010-05-19 14:45:46

2009-11-11 13:49:02

路由器技術(shù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: www.一区二区 | 日韩一级欧美一级 | 亚洲午夜精品一区二区三区他趣 | 综合色久| 亚洲精品在线视频 | 精品亚洲一区二区三区四区五区 | 日韩在线观看网站 | 亚洲成人精品 | 欧美中文字幕一区二区 | 国产成人网 | 九一在线观看 | av网站免费观看 | 日韩一区二区三区视频在线观看 | 一本色道久久综合亚洲精品高清 | 狠狠ri | 国产精品久久久久久久久久久久 | 成人av一区二区三区 | 亚洲a一区二区 | 国产伦一区二区三区久久 | 国产精品欧美精品日韩精品 | 羞羞视频在线观看 | 一a级片| 精国产品一区二区三区 | 久久精品国产一区二区电影 | 中文字幕不卡一区 | 久久精品一区二区三区四区 | 日韩欧美高清 | 欧美日韩在线视频一区 | 一区二区不卡 | 91热在线| 欧美日韩国产免费 | 亚洲二区在线 | 九九热免费视频在线观看 | 久久99精品久久久久久青青日本 | 美女国产| 国产av毛片| 免费在线一区二区三区 | 婷婷综合激情 | 亚洲国产成人精 | 亚洲综合久久久 | 日日天天 |