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
C和C++之間庫的互相調用
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動態庫:
#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++程序去調用它:
#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++庫進行二次封裝:
#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++庫:
#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++庫中含有類的,可以在二次接口函數中生成臨時對象來調用對應的功能函數,當然要根據實際情況來定了。
c/c++動態庫