1. 程式人生 > >Linux連結庫三(C跟C++之間動態庫的相互呼叫)

Linux連結庫三(C跟C++之間動態庫的相互呼叫)

http://www.cppblog.com/wolf/articles/74928.html

http://www.cppblog.com/wolf/articles/77828.html

http://www.jb51.net/article/34990.htm

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):
[[email protected] test]# gcc --shared -o libhello.so hello.c
[[email protected] 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;
}


編譯並執行:
[[email protected] test]# g++ test.cpp -o test -lhello
[[email protected] test]# ./test
hello
[[email protected] 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到系統庫目錄下:
[[email protected] test]# g++ --shared -o libworld.so world.cpp
[[email protected] 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到系統庫目錄下:
[[email protected] test]# g++ --shared -o libmid.so mid.cpp -lworld
[[email protected] test]# cp libmid.so /lib/
3).C程式通過連結二次介面庫去呼叫C++庫:

// test.c

#include <stdio.h>

int main()
{
  m_world();

  return 0;
}


編譯並執行:
[[email protected] test]# gcc test.c -l mid -o test
[[email protected] test]# ./test
world
[[email protected] test]# 
注:如果對於C++庫中含有類的,可以在二次介面函式中生成臨時物件來呼叫對應的功能函式,當然要根據實際情況來定了。

相關推薦

工程呼叫vs2015中兩個工程,相互呼叫

任務:想在try工程下,呼叫另一個工程speech_emotion的cpp函式 同一解決方案下,可以有多個工程相互呼叫,具體是要注意以下幾個內容: 1、呼叫speech_emotion中的.h檔案,需要增加路徑的引用,找到speech_emotion下的

Linux連結CC++之間動態相互呼叫

http://www.cppblog.com/wolf/articles/74928.html http://www.cppblog.com/wolf/articles/77828.html http://www.jb51.net/article/34990.htm extern "C"的理解: 很多人認

單目跟蹤位姿產品研發----在linux下將c++工程打包成動態so檔案API

       單目跟蹤位姿專案由對方公司提供應用場景,我方研發核心演算法,通過c++實現功能,然後對方公司通過java\js來開發炫酷的介面,共同合作完成,最終對方公司負責銷售推廣,推向市場。因此,涉及到java介面呼叫c++核心程式碼的問題。 現記錄

AES加密演算法原始碼c++版其實c版差不多

打包下載 //============================================================================ // 檔名: AES.h (c++) // 作者    : 幽靈劍客 // 版本    : 2008021

C++基礎知識之動態靜態

www lar 而在 lib ont 運行程序 gpa prope 否則 一、 靜態庫與動態庫 庫(library),一般是一種可執行的二進制格式,被操作系統載入內存執行。 我們通常把一些公用函數制作成函數庫,供其它程序使用。函數庫分為靜態庫和動態庫 靜態庫和動態庫區

C# 配置引用程式集的路徑分離exe和dll從指定路徑呼叫

問題:在開發較大的專案時,會引用很多其他專案的dll,而我們規劃將不同型別的dll放在不同的功能目錄下,此時我們通過反射動態載入時,出現System.IO.FileNotFoundException: 未能載入檔案或程式集的問題。 程式集DLL分為兩類: 2)私有DL

NDK基礎java ,c/c++, jni之間的關係及java和c/c++之間相互呼叫

1.java,c/c++,和jni之間的關係 java和c/c++可以相互呼叫,是因為java虛擬機器中的JNI。簡單的說就是用c/c++編寫一個動態連結庫讓Java虛擬機器去呼叫。(在windows環境下動態連結庫就是.dll檔案, 在Linux下就是.so檔案) 2.

Linux curses 總結對鍵盤的操作

cbreak & cooked(預處理模式) curses程式的控制模式函式 curses程式的鍵盤操作函式 小栗子 curses程式的鍵盤操作是對底層的簡單封裝介面 cbreak & cooked(預處理模式) cooked:

c++呼叫c語言的so動態

1.環境  ubuntu14.04   gcc 4.8.4 g++ 4.8.4 2.庫檔案 1)原始碼 //clTest.c int add(int a, int b) { return a + b; } 2)生成庫檔案 cd到clTest.c所

Linux學習筆記常用命令

11 、 壓縮檔案命令:gzip 命令名稱:gzip  命令英文原意:GNU zip  命令所在路徑:/bin/gzip  執行許可權:所有使用者  語法:gzip [檔案]  -d 解壓縮 同gunzip 功能描述:壓縮檔案 不保留原檔案,只剩壓縮包 壓縮後文件格式:.gz 12、解壓縮.gz的壓縮檔案命令

C 語言兩個動態函式重名問題

應用程式a(a.c),動態庫liba.so(liba.h, liba.c),libb.so,均實現了func() gcc -la -lb a.c 則呼叫的是liba.so中的函式實現 gcc -lb -la a.c 則呼叫的是libb.so中的函式實現

JSP編程專題2之JSP核心自定義EL函數和標簽

自定義el函數 jstl中的el函數 自定義el標簽 自定義帶屬性的el標簽 自定義el遍歷標簽 1、自定義EL函數:a、自定義類,類中的方法使用static修飾符修飾:b、該類以及函數,需要在一個擴展名為.tld(tld,tag library definition,標簽庫定義)的XML

solr學習測試類,含普通與ExtractingRequestHandler測試

list clas map ber -h div comment test bst solr客戶端基本是配置出來的,服務端可以對其進行測試,我使用的是solrj服務端。 如果初學solr,先使用普通的測試類: Java代碼 import java.io.IO

使程式在Linux下後臺執行 關掉終端繼續讓程式執行的方法

一、為什麼要使程式在後臺執行 我們計算的程式都是週期很長的,通常要幾個小時甚至一個星期。我們用的環境是用putty遠端連線到日本Linux伺服器。所以使程式在後臺跑有以下三個好處: 1:我們這邊是否關機不影響日本那邊的程式執行。(不會像以前那樣,我們這網路一斷開,或一關機,程式就

Linux作業系統實驗一在VMware安裝Red Hat Enterprise 6.X

開啟VMware——建立新的虛擬機器————典型(推薦)——下一步 安裝程式光碟映像檔案——開啟系統所在的映像檔案——下一步 輸入系統的名稱和自己的使用者名稱 密碼——下一步 輸入虛擬機器的名稱和位置——下一步 輸入合適的最大磁碟大小

初識面向物件經典類/多型/鴨子型別/初識封裝

py2中的經典類 coding:utf-8 class D: pass # def func(self): # print('d')class B(D): pass # def func(self): # print('b')class C(D):

Linux權限管理用戶、組、文件管理

參數 id號 逗號 獨立 gpasswd pip who 結束 目錄 一、 Linux上的文件管理類命令都有哪些,其常用的使用方法及其相關示例演示。 1、 文件查看類命令cat,tac, head, tail, more, less, ls ,file: -ls : lis

linux關於執行編譯後命令找不到動態.so檔案的解決辦法。

方法一:(沒有root許可權)利用find找到報錯動態庫檔案目錄lib路徑,vim /home/xxx/.bashrc 新增一行: export LD_LIBRARY_PATH=動態庫/lib:$LD_LIBRARY_PATH 然後 source /home/xxx/.bashrc

Linux----網路程式設計I/O複用之select系統呼叫

io_select_ser.c 1. #include <string.h> 2. #include <assert.h> 3. #include <unistd.h> 4. #include <stdio.h> 5. #in

Linux 操作命令 cp忙了一陣,特來更新 一下

mtime 多個 文件比較 ber 更新 源文件 指定權限 clob ORC 1、簡介     cp 命令是 copy 的縮寫     2、主要功能