• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            把握命運,追逐夢想

            對自己所做的事要有興趣,同時還要能夠堅持不懈

            統(tǒng)計

            留言簿(1)

            閱讀排行榜

            評論排行榜

            extern "C" __declspec(dllexport) __declspec(dllimport) 和 def

            前面的extern "C"  __declspec(dllexport)  __declspec(dllimport)都是用于函數(shù)或者變量,甚至類的聲明的(可以把extern "C"放在class的前面,但是編譯器會忽略掉,最后產(chǎn)生的還是C++修飾符,而不是C修飾符)這樣的用法有個好處就是下面的代碼可以在混有類的函數(shù)和變量上使用下面的宏,雖然對類不起作用:
            #ifdef __cplusplus
            extern "C"
            {
            //函數(shù)聲明
            //變量聲明,變量一般前面都有extern
            //類聲明,這個不起作用,編譯器直接忽略掉class前面的extern “C”
            #ifdef __cplusplus
            }

            #endif
            C 和C++ 對應(yīng)不同的調(diào)用約定,產(chǎn)生的修飾符也各不相同,如下:
            調(diào)用約定 extern "C" 或 .c 文件 .cpp、.cxx 或 /TP

            C 命名約定 (__cdecl)

            _test

            ?test@@ZAXXZ

            Fastcall 命名約定 (__fastcall)

            @test@0

            ?test@@YIXXZ

            標(biāo)準(zhǔn)調(diào)用命名約定 (__stdcall)

            _test@0

            ?test@@YGXXZ


            __declspec(dllexport)  __declspec(dllimport)一般也是使用宏的形式:
            #ifdef ONEDLL_EXPORTS
            #define ONEDLL_API __declspec(dllexport)
            #else
            #define ONEDLL_API __declspec(dllimport)
            #endif
            這樣在DLL代碼本身就是__declspec(dllexport) ,在使用DLL的程序中就變成了__declspec(dllimport),這兩標(biāo)志分別用來指明當(dāng)前的函數(shù)將被導(dǎo)出,和當(dāng)前函數(shù)是被導(dǎo)入的。
             

            上面的兩個宏結(jié)合一下就是下面這樣的了:
            // 下列 ifdef 塊是創(chuàng)建使從 DLL 導(dǎo)出更簡單的
            // 宏的標(biāo)準(zhǔn)方法。此 DLL 中的所有文件都是用命令行上定義的 ONEDLL_EXPORTS
            // 符號編譯的。在使用此 DLL 的
            // 任何其他項目上不應(yīng)定義此符號。這樣,源文件中包含此文件的任何其他項目都會將
            // ONEDLL_API 函數(shù)視為是從 DLL 導(dǎo)入的,而此 DLL 則將用此宏定義的
            // 符號視為是被導(dǎo)出的。
            #ifdef ONEDLL_EXPORTS
            #define ONEDLL_API __declspec(dllexport)
            #else
            #define ONEDLL_API __declspec(dllimport)
            #endif

            // 此類是從 OneDll.dll 導(dǎo)出的
            #ifdef __cplusplus
            extern "C"
            {
            #endif
            class ONEDLL_API COneDll {
            public:
                COneDll(
            void);
                
            ~COneDll(void);
                
                
            // TODO: 在此添加您的方法。
                int m_a;
                
            int m_b;
                
            int *m_p;
                
            int m_n;

                
            void AddValue();

            }
            ;

            extern ONEDLL_API int nOneDll;

            ONEDLL_API 
            int fnOneDll(void);

            #ifdef __cplusplus
            }

            #endif

            如果調(diào)用模塊和被調(diào)用模塊都是C++(而且是同一種編成環(huán)境,如VC,甚至需要同一版本的VC),那么就不需要extern “C”了,因為這個標(biāo)志的作用就是用在函數(shù)和變量聲明前,無論是調(diào)用模塊,還是被調(diào)用模塊,都將生成C修飾符,調(diào)用模塊將需要C修飾符的函數(shù),而被調(diào)用模塊將產(chǎn)生C修飾符的函數(shù),所以這個標(biāo)志在兩者都是C++的時候使用并不受影響,不使用這個標(biāo)志,也不受影響。
            但是如果C模塊要調(diào)用C++ 模塊,那么C++模塊就需要使用extern “C”,當(dāng)然C不用,由于是在頭文件的聲明中使用,所以使用下面的宏能夠使得這個頭文件也在C中順利使用:
            #ifdef __cplusplus
            extern "C"
            {
            //函數(shù)聲明
            //變量聲明,變量一般前面都有extern
            //類聲明,這個不起作用,編譯器直接忽略掉class前面的extern “C”
            #ifdef __cplusplus
            }

            #endif

            如果C++模塊要調(diào)用C模塊,那么C++模塊還是需要extern “C”,當(dāng)然C不用,由于是在頭文件的聲明中使用,所以使用上面的宏同樣能夠使得這個頭文件也在C中順利使用。

            總結(jié)一下就是加上extern “C”在什么情況下都沒錯,但是要注意函數(shù)重載的問題。



            def文件是一種比較麻煩的方法,下面是MSDN中的部分內(nèi)容:
             

            模塊定義 (.def) 文件是包含一個或多個描述 DLL 各種屬性的 Module 語句的文本文件。如果不使用 __declspec(dllexport) 關(guān)鍵字導(dǎo)出 DLL 的函數(shù),則 DLL 需要 .def 文件。

            .def 文件必須至少包含下列模塊定義語句:
            1.文件中的第一個語句必須是 LIBRARY 語句。此語句將 .def 文件標(biāo)識為屬于 DLL。LIBRARY 語句的后面是 DLL 的名稱。鏈接器將此名稱放到 DLL 的導(dǎo)入庫中。
            2.EXPORTS 語句列出名稱,可能的話還會列出 DLL 導(dǎo)出函數(shù)的序號值。通過在函數(shù)名的后面加上 @ 符和一個數(shù)字,給函數(shù)分配序號值。當(dāng)指定序號值時,序號值的范圍必須是從 1 到 N,其中 N 是 DLL 導(dǎo)出函數(shù)的個數(shù)。

            例如,包含實現(xiàn)二進制搜索樹的代碼的 DLL 看上去可能像下面這樣:

            LIBRARY   BTREE
            EXPORTS
               Insert   @
            1
               Delete   @
            2
               Member   @
            3
               Min   @
            4

            提示:

            如果希望優(yōu)化 DLL 文件的大小,請對每個導(dǎo)出函數(shù)使用 NONAME 屬性。使用 NONAME 屬性時,序號存儲在 DLL 的導(dǎo)出表中而非函數(shù)名中。如果導(dǎo)出許多函數(shù),這樣做可以節(jié)省相當(dāng)多的空間。


            其實__declspec(dllexport)的作用就是讓編譯器按照某種預(yù)定的方式(前面大致解釋了這種方式的規(guī)則)來輸出導(dǎo)出函數(shù)及變量的符號,而def文件則是自己為每一個函數(shù)和變量指定導(dǎo)出符號,所以def是一個非自動化,手工很強的方式,不是特殊情況的話,實在沒有必要浪費這些時間。
            還有一個問題,就是使用def會把調(diào)用方式和__declspec(dllexport)的作用全部覆蓋掉,所以還需要自己處理調(diào)用方式不同產(chǎn)生的錯誤。
            一般使用def文件的情況是你需要使用運行時加載,并且需要使用GetProcAddress函數(shù)獲得函數(shù)地址,這個函數(shù)需要直接指明函數(shù)產(chǎn)生的導(dǎo)出符號,而可以自己指定導(dǎo)出符號的方式就是使用def。
            def文件的具體語法可以看看msdn。

            posted on 2009-08-03 17:12 把握命運 閱讀(18669) 評論(0)  編輯 收藏 引用


            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            天天躁日日躁狠狠久久| 精品无码久久久久久尤物| 九九久久精品无码专区| 欧美粉嫩小泬久久久久久久 | 久久国产成人午夜AV影院| 亚洲国产精品18久久久久久| 久久超碰97人人做人人爱| 99久久国产综合精品五月天喷水 | 欧美久久精品一级c片片| 国产日韩久久久精品影院首页| 成人国内精品久久久久影院VR| 99久久精品国产麻豆| 久久久精品国产亚洲成人满18免费网站| 久久91这里精品国产2020| 国产精品激情综合久久| 伊人久久综合成人网| 97精品国产97久久久久久免费| 99久久国产宗和精品1上映| 久久久久国产精品| 丰满少妇人妻久久久久久| 日韩久久无码免费毛片软件| 久久久久久亚洲精品成人| 久久受www免费人成_看片中文| 久久精品成人欧美大片| 亚洲va久久久噜噜噜久久男同| 欧美激情精品久久久久久久| 久久精品人人做人人爽电影蜜月 | 无码AV波多野结衣久久| 久久激情五月丁香伊人| 久久精品成人免费网站| 久久婷婷国产综合精品 | 国内精品久久久久伊人av| 亚洲一级Av无码毛片久久精品| 国产—久久香蕉国产线看观看| 久久久精品2019免费观看| 欧美精品国产综合久久| 欧美精品一区二区久久| 久久播电影网| 国产真实乱对白精彩久久| 国产AV影片久久久久久| 国产伊人久久|