__declspec(dllimport)的作用

pamxy發表於2013-03-24

轉自:http://blog.csdn.net/mniwc/article/details/7993361

是時候總結一下__declspec(dllimport)的作用了。可能有人會問:__declspec(dllimport)和__declspec(dllexport)是一對的,在動態連結庫中__declspec(dllexport)管匯出,__declspec(dllimport)管匯出,就像一個國家一樣,有出口也有進口,有什麼難理解的呢?這是一種很自然的思路,開始我也是這樣理解。


但是在兩年前的一個專案中,我發現不用__declspec(dllimport)似乎也可以。比如現在我新建一個使用共享MFC DLL的規則DLL工程:DllDlg。然後我新建兩個檔案:DllApi.h和DllApi.cpp。DllApi.h作為介面文 件,DllApi.cpp作為實現檔案。


接著在DllApi.h宣告一個函式:

  1. __declspec(dllexportvoid HelloWorld();
  1. __declspec(dllexportvoid HelloWorld();  


在DllApi.cpp寫這個函式的實現:


  1. void HelloWorld()
  2. {
  3. AfxMessageBox(_T("HelloWorld"));
  4. }
  1. void HelloWorld()  
  2. {  
  3.     AfxMessageBox(_T("HelloWorld"));  
  4. }  


這樣外部的應用程式或dll就能呼叫HelloWorld函式。這裡要特別提醒的是:有些網友說要把DllApi.h中的__declspec(dllexport) void HelloWorld();改為__declspec(dllimport) void HelloWorld();才能提供給外部呼叫,實際上這並不需要,這個我已經測試過。從那時我就產生一個疑問:照這樣,像類似下面的:


  1. #ifdef _EXPORTING
  2. #define API_DECLSPEC __declspec(dllexport)
  3. #else
  4. #define API_DECLSPEC __declspec(dllimport)
  5. #endif
  1. #ifdef _EXPORTING  
  2. #define API_DECLSPEC    __declspec(dllexport)  
  3. #else  
  4. #define API_DECLSPEC    __declspec(dllimport)  
  5. #endif  

是不是就只剩下一種作用:讓外部呼叫者看得更自然些,知道哪些介面是自己工程需要匯入的?__declspec(dllimport)是不是一點實際作用都沒有呢?這個疑問一直盤旋在我的腦海。直到最近,我在CSDN論壇上發了一個帖子:


__declspec(dllimport) 的作用到底在哪裡呢?

總結了各位大蝦的發言,特得出如下結論:
1. 在匯入動態連結庫中的全域性變數方面起作用:
使用類似

  1. #ifdef _EXPORTING
  2. #define API_DECLSPEC __declspec(dllexport)
  3. #else
  4. #define API_DECLSPEC __declspec(dllimport)
  5. #endif
  1. #ifdef _EXPORTING  
  2. #define API_DECLSPEC    __declspec(dllexport)  
  3. #else  
  4. #define API_DECLSPEC    __declspec(dllimport)  
  5. #endif  


可以更好地匯出dll中的全域性變數,比如按照的巨集,可以在dll中這樣匯出全域性變數:


  1. API_DECLSPEC CBtt g_Btt;
  1. API_DECLSPEC CBtt g_Btt;  


然後在呼叫程式這樣匯入:


  1. API_DECLSPEC CBtt g_Btt;
  1. API_DECLSPEC CBtt g_Btt;  


當然也可以使用extern關鍵字,比如在dll中這樣匯出全域性變數:


  1. CBtt g_Btt;
  1. CBtt g_Btt;  


然後在呼叫程式這樣匯入:


  1. extern CBtt g_Btt;
  1. extern CBtt g_Btt;  


但據說使用__declspec(dllimport)更有效。


2. __declspec(dllimport)的作用主要體現在匯出類的靜態成員方面,
比如在動態連結庫中定義這樣一個匯出類:


  1. class __declspec(dllexport) CBtt
  2. {
  3. public:
  4. CBtt(void);
  5. ~CBtt(void);
  6. public:
  7. CString m_str;
  8. static int GetValue()
  9. {
  10. return m_nValue;
  11. }
  12. private:
  13. static int m_nValue;
  14. };
  1. class __declspec(dllexport) CBtt  
  2. {  
  3. public:  
  4.     CBtt(void);  
  5.     ~CBtt(void);  
  6. public:  
  7.     CString m_str;  
  8.     static int GetValue()  
  9.     {  
  10.         return m_nValue;  
  11.     }  
  12. private:  
  13.     static int m_nValue;  
  14. };  


照上面這樣宣告,外部雖然可以使用CBtt類,但不能使用CBtt類的GetValue函式,一使用就會出現無法解析的外部符號 "public: static int CBtt::m_nValue" (?m_nValue@CBtt@@2HA)。只有如下宣告才能使用CBtt類的GetValue函式:


  1. #ifdef _EXPORTING
  2. #define API_DECLSPEC __declspec(dllexport)
  3. #else
  4. #define API_DECLSPEC __declspec(dllimport)
  5. #endif
  6. class API_DECLSPEC CBtt
  7. {
  8. public:
  9. CBtt(void);
  10. ~CBtt(void);
  11. public:
  12. CString m_str;
  13. static int GetValue()
  14. {
  15. return m_nValue;
  16. }
  17. private:
  18. static int m_nValue;
  19. };
  1. #ifdef _EXPORTING  
  2. #define API_DECLSPEC    __declspec(dllexport)  
  3. #else  
  4. #define API_DECLSPEC    __declspec(dllimport)  
  5. #endif  
  6. class API_DECLSPEC CBtt  
  7. {  
  8. public:  
  9.     CBtt(void);  
  10.     ~CBtt(void);  
  11. public:  
  12.     CString m_str;  
  13.     static int GetValue()  
  14.     {  
  15.         return m_nValue;  
  16.     }  
  17. private:  
  18.     static int m_nValue;  
  19. };  



3. 使用隱式使用dll時,不加__declspec(dllimport)完全可以,使用上沒什麼區別,只是在生成的二進位制程式碼上稍微有點效率損失。


a、 不加__declspec(dllimport)時,在使用dll中的函式時,編譯器並不能區別這是個普通函式,還是從其它dll裡匯入的函式,所以其生 成的程式碼如下:

call 地址1

地址1:
jmp 實際函式地址


b、有 __declspec(dllimport)時,編譯器知道這是要從外部dll匯入的函式,從而在生成的exe的輸入表裡留有該項,以便在執行 exe,PE載入器載入exe時對輸入地址表IAT進行填寫,這樣生成的程式碼如下:

call dword ptr[輸入表裡哪項對應的記憶體地址] (注意:現在就不需要jmp stub了)。這裡
有興趣的朋友可以參看《編譯原理》和 PE檔案格式。


4.使用__declspec(dllimport)體現了語言的一種對稱美,比如雖然!true就是表示false,但是我們還是需要false這個關鍵字,這裡體現了一種對稱美。

在此特別感謝CSDN的眾位大俠:superdiablo、wltg2001、ccpaishi、jszj、WizardK、hurryboylqs、jingzhongrong、jameshooo、glacier3d、winnuke、starnight1981、laiyiling、yang79tao、ForestDB、zhouzhipen、lxlsymbome、Beyond_cn。


參考文獻:


1. __declspec(dllimport) 到底有什麼用?



from: http://blog.csdn.net/clever101/article/details/5421782

上一篇:VS2005設定Release模式可除錯

相關文章