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

            Mike's blog

              C++博客 :: 首頁 :: 聯系 :: 聚合  :: 管理
              0 Posts :: 23 Stories :: 83 Comments :: 0 Trackbacks

            常用鏈接

            留言簿(17)

            我參與的團隊

            搜索

            •  

            最新評論

            昨晚有個朋友問我關于在C中調用C++庫的問題,今天午飯后,由于脖子痛的厲害而沒有加入到我們組的“每天一戰”的行列中去,所以正好將C和C++之間的庫調用關系做個總結。

            1.extern "C"的理解:
            很多人認為"C"表示的C語言,實際并非如此,"C"表示的是一種鏈接約定,只是因C和C++語言之間的密切關系而在它們之間更多的應用而已。實際上Fortran和匯編語言也常常使用,因為它們也正好符合C實現的約定。
            extern "C"指令描述的是一種鏈接約定,它并不影響調用函數的定義,即時做了該聲明,對函數類型的檢查和參數轉換仍要遵循C++的標準,而不是C。

            2.extern "C"的作用:
            不同的語言鏈接性是不同的,那么也決定了它們編譯后的鏈接符號的不同,比如一個函數void fun(double d),C語言會把它編譯成類似_fun這樣的符號,C鏈接器只要找到該函數符號就可以鏈接成功,它假設參數類型信息是正確的。而C++會把這個函數編譯成類似_fun_double或_xxx_funDxxx這樣的符號,在符號上增加了類型信息,這也是C++可以實現重載的原因。
            那么,對于用C編譯器編譯成的庫,用C++直接鏈接勢必會出現不能識別符號的問題,是的,需要extern "C"的時刻來了,它就是干這個用的。extern "C" 的作用就是讓編譯器知道要以C語言的方式編譯和連接封裝函數。

            3.在C++中調用C庫的例子:
            1).做一個C動態庫:

            // hello.c:

            #include 
            <stdio.h>

            void hello()
            {
              printf(
            "hello\n");
            }

             

            編譯并copy到系統庫目錄下(也可以自己定義庫目錄,man ldconfig):
            [root@coredump test]# gcc --shared -o libhello.so hello.c
            [root@coredump test]# cp libhello.so /lib/
            2).寫個C++程序去調用它:

            // test.cpp

            #include 
            <iostream>

            #ifdef __cplusplus
            extern "C" {               // 告訴編譯器下列代碼要以C鏈接約定的模式進行鏈接
            #endif

            void hello();

            #ifdef __cplusplus
            }
            #endif

            int main()
            {
              hello();

              
            return 0;
            }


            編譯并運行:
            [root@coredump test]# g++ test.cpp -o test -lhello
            [root@coredump test]# ./test
            hello
            [root@coredump test]#
            3).__cplusplus宏的條件編譯:
            為什么要加這個條件編譯呢?小沈陽有話:小妹,這是為什么呢?
            因為這種技術也可能會用在由C頭文件產生出的C++文件中,這樣使用是為了建立起公共的C和C++文件,也就是保證當這個文件被用做C文件編譯時,可以去掉C++結構,也就是說,extern "C"語法在C編譯環境下是不允許的。
            比如:將上面的test.cpp更名為test.c,將頭文件改為stdio.h,將條件編譯去掉,再用gcc編譯就可以看到效果。而即使做了上面的修改,如果用g++編譯就可以正常使用,這就是我上面說的“公共的C和C++文件”的意思。

            4.C調用C++庫:
            C++調用C庫看上去也不是那么困難,因為C++本身就有向前(向C)兼容的特性,再加上純天然的extern "C"約定,使得一切都是那么自然。而讓C調用C++的庫似乎就沒那么容易,不過也不是不可以的。
            說到這里我得休息一下,大中午的,出去抽根煙先,不過我也相信如果你不知道答案,看到這里的時候肯定在到處找板磚,恨不得敲開我的腦殼子。我能理解,我也習慣了,我有個學姐一看到我第一反應就是扔出一塊磚頭先!
            言歸正傳,還是要借助這純天然的extern "C"。

            1)做一個C++庫:

            // world.cpp

            #include 
            <iostream>

            void world()
            {
              std::cout 
            << "world" << std::endl;
            }


            編譯并copy到系統庫目錄下:
            [root@coredump test]# g++ --shared -o libworld.so world.cpp
            [root@coredump test]# cp libworld.so /lib/
            2)做一個中間接口庫,對C++庫進行二次封裝:

            // mid.cpp

            #include 
            <iostream>

            void world();

            #ifdef __cplusplus
            extern "C" {  // 即使這是一個C++程序,下列這個函數的實現也要以C約定的風格來搞!
            #endif

              
            void m_world()
              {
                world();
              }

            #ifdef __cplusplus
            }
            #endif


            其中方法m_world即為libworld庫中world方法的二次封裝,編譯并copy到系統庫目錄下:
            [root@coredump test]# g++ --shared -o libmid.so mid.cpp -lworld
            [root@coredump test]# cp libmid.so /lib/
            3).C程序通過鏈接二次接口庫去調用C++庫:

            // test.c

            #include 
            <stdio.h>

            int main()
            {
              m_world();

              
            return 0;
            }


            編譯并運行:
            [root@coredump test]# gcc test.c -l mid -o test
            [root@coredump test]# ./test
            world
            [root@coredump test]#
            注:如果對于C++庫中含有類的,可以在二次接口函數中生成臨時對象來調用對應的功能函數,當然要根據實際情況來定了。

            5.靠,過了午休時間了,收工!

            posted on 2009-03-25 14:34 老狼 閱讀(20792) 評論(6)  編輯 收藏 引用 所屬分類: C/C++

            Feedback

            # re: C和C++之間庫的互相調用[未登錄] 2012-03-24 23:10 roger
            寫的太好了,多謝啊  回復  更多評論
              

            # re: C和C++之間庫的互相調用 2012-10-16 13:49 connie
            交代的好清楚,這次我看懂了  回復  更多評論
              

            # re: C和C++之間庫的互相調用 2013-08-08 15:43
            如果做成靜態庫會怎么樣呢?  回復  更多評論
              

            # re: C和C++之間庫的互相調用 2013-11-20 09:28 藝搜天下
            總結得不錯,支持下。
            by www.elesos.com 站長  回復  更多評論
              

            # re: C和C++之間庫的互相調用 2015-08-28 10:36 藝搜天下
            C++庫中含有類的情況希望能總結下哈,已轉載  回復  更多評論
              

            # re: C和C++之間庫的互相調用 2016-01-06 12:46 maxwell
            按照例子做在C++ 中調用C的動態庫出現 ”undfined reference to 'hello()'的錯誤,在C中調用C++出現 undefined reference to 'm_world'的錯誤,請問博主如何解決?  回復  更多評論
              

            综合人妻久久一区二区精品| 国色天香久久久久久久小说| 久久久久久国产精品无码下载 | 久久久久久a亚洲欧洲aⅴ| 香蕉久久一区二区不卡无毒影院| 久久综合鬼色88久久精品综合自在自线噜噜| 精品国产乱码久久久久久1区2区| 亚洲欧美国产精品专区久久| 久久精品国产精品亚洲艾草网美妙 | 无码精品久久久久久人妻中字| 久久婷婷五月综合成人D啪| 99久久er这里只有精品18| 久久精品无码专区免费| 亚洲精品视频久久久| a级成人毛片久久| 精品伊人久久大线蕉色首页| 日本一区精品久久久久影院| 无码乱码观看精品久久| 久久99精品久久久久久齐齐| 国内精品久久久久久久涩爱| 少妇内射兰兰久久| 亚洲欧洲久久久精品| 精品久久久久久成人AV| 久久青青色综合| 久久国产精品偷99| 久久综合九色综合欧美狠狠| 久久精品麻豆日日躁夜夜躁| 色成年激情久久综合| 人妻久久久一区二区三区| 久久亚洲国产欧洲精品一| 日韩精品无码久久久久久| 久久精品卫校国产小美女| 日本亚洲色大成网站WWW久久 | 人人狠狠综合久久亚洲| 色噜噜狠狠先锋影音久久| 91精品国产色综合久久| 久久天天躁狠狠躁夜夜96流白浆| 亚洲级αV无码毛片久久精品 | 久久亚洲精品国产亚洲老地址| 国产成人无码精品久久久免费 | 亚洲午夜久久久影院|