• <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>

            麒麟子

            ~~

            導航

            <2009年3月>
            22232425262728
            1234567
            891011121314
            15161718192021
            22232425262728
            2930311234

            統計

            常用鏈接

            留言簿(12)

            隨筆分類

            隨筆檔案

            Friends

            WebSites

            積分與排名

            最新隨筆

            最新評論

            閱讀排行榜

            評論排行榜

            [轉] extern "C"

            前些天,編程序是用到了很久以前寫的C程序,想把里面的函數利用起來,連接發現出現了找不到具體函數的錯誤:

            以下是假設舊的C程序庫

            C的頭文件

            /*-----------c.h--------------*/
            #ifndef _C_H_
            #define _C_H_
            extern int add(int x, int y);
            #endif

            C的源文件

            /*-----------c.c--------------*/
            int
            add(int x, int y){
            return
            x+y;
            }

            C++的調用

            /*-----------cpp.cpp--------------*/
            #include "c.h"
            void main()
            {

            add(1, 0);
            }

            這樣編譯會產生錯誤cpp.obj : error LNK2001: unresolved external symbol "int __cdecl add(int,int)" (?add@@YAHHH@Z),原因是找不到add的目標模塊

            這才令我想起C++重載的函數命名方式和C函數的命名方式,讓我們回顧一下:C中函數編譯后命名會在函數名前加以"_",比如add函數編譯成obj文件時的實際命名為_add,而c++命名則不同,為了實現函數重載同樣的函數名add因參數的不同會被編譯成不同的名字

            例如

            int add(int , int)==>add@@YAHHH@Z,

            float add(float , float )==>add@@YAMMM@Z,

            以上是VC6的命名方式,不同的編譯器會不同,總之不同的參數同樣的函數名將編譯成不同目標名,以便于函數重載是調用具體的函數。

            編譯cpp.cpp中編譯器在cpp文件中發現add(1, 0);的調用而函數聲明為extern int add(int x, int y);編譯器就決定去找add@@YAHHH@Z,可惜他找不到,因為C的源文件把extern int add(int x, int y);編譯成_add了;

            為了解決這個問題C++采用了extern "C",這就是我們的主題,想要利用以前的C程序庫,那么你就要學會它,我們可以看以下標準頭文件你會發現,很多頭文件都有以下的結構

            #ifndef __H
            #define __H
            #ifdef __cplusplus
            extern "C" {
            #endif

            extern
            int f1(int, int);
            extern
            int f2(int, int);
            extern
            int f3(int, int);


            #ifdef __cplusplus
            }
            #endif

            #endif /*__H*/

            如果我們仿制該頭文件可以得到

            #ifndef _C_H_
            #define _C_H_
            #ifdef __cplusplus
            extern "C" {
            #endif

            extern
            int add(int, int);

            #ifdef __cplusplus
            }
            #endif

            #endif /* _C_H_ */

            這樣編譯

            /*-----------c.c--------------*/
            int
            add(int x, int y){
            return
            x+y;
            }

            這時源文件為*.c,__cplusplus沒有被定義,extern "C" {}這時沒有生效對于C他看到只是extern int add(int, int);
            add函數編譯成_add(int, int);

            而編譯c++源文件

            /*-----------cpp.cpp--------------*/
            #include "c.h"
            void main()
            {

            add(1, 0);
            }

            這時源文件為*.cpp,__cplusplus被定義,對于C++他看到的是extern "C" {extern int add(int, int);}編譯器就會知道 add(1, 0);調用的C風格的函數,就會知道去c.obj中找_add(int, int)而不是add@@YAHHH@Z

            這也就為什么DLL中常看見extern "C" {},windows是采用C語言編制他首先要考慮到C可以正確調用這些DLL,而用戶可能會使用C++而extern "C" {}就會發生作用

            posted on 2009-05-17 23:46 麒麟子 閱讀(308) 評論(0)  編輯 收藏 引用 所屬分類: Programming

            精品一二三区久久aaa片| 亚洲国产成人精品女人久久久| 一本大道久久香蕉成人网| 久久一区二区三区99| 亚洲精品国产综合久久一线| 中文字幕久久久久人妻| 久久66热人妻偷产精品9| 国产精品伦理久久久久久| 久久久精品国产| 99久久精品国产麻豆| 日批日出水久久亚洲精品tv| 久久久老熟女一区二区三区| 国产免费久久久久久无码| 亚洲日本久久久午夜精品| 国内精品久久久久久99蜜桃| 人人狠狠综合久久亚洲| 国产精品久久久久久久| 模特私拍国产精品久久| 夜夜亚洲天天久久| 久久久久亚洲精品天堂| 久久无码专区国产精品发布| 久久免费线看线看| 久久精品夜夜夜夜夜久久| 久久婷婷人人澡人人| 精品一区二区久久| 九九久久自然熟的香蕉图片| 一本色道久久综合狠狠躁| 午夜精品久久影院蜜桃| 久久精品国产精品亚洲下载| 欧美一区二区精品久久| 色综合久久久久无码专区| 一97日本道伊人久久综合影院| 久久黄色视频| 性做久久久久久免费观看| 久久狠狠一本精品综合网| 精品久久久久久国产牛牛app| 久久91精品久久91综合| 精品久久久久久| 99热成人精品免费久久| 久久精品无码专区免费| 国内精品伊人久久久久网站|