本文转自http://blogold.chinaunix.net/u/18517/showart_196363.html
C++中的extern C
本文主要讨论C++中extern C的使用。 作者:tyc611, 2006-11-06
在使用extern "C"声明时,有两种不同的形式:一是extern "C"后跟函数(或者变量)声明;另外就是extern "C"{函数(或变量)声明}。第一种形式(inline form)表示同时声明为外部连接和C风格连接;而第二种形式仅声明为C风格连接(并不附带外部声明)。例如:
//第一种形式 extern "C" int foo; //外部变量声明并且C风格连接 extern "C" void bar(); //外部函数声明并且C风格连接,此时就不能为static
//第二种形式 extern "C" { int foo; //C风格连接,并且定义变量(不是外部连接声明!) void bar(); }
所以当将第二种形式改写成如下形式时,两种形式才具有等价性:
//第二种形式修改 extern "C" { extern int foo; //这儿的extern声明其作用对象为外部连接 extern void bar(); }
对于有extern "C"声明的函数,其作用只是改变了函数的连接形式(采用C风格,即在目标文件中的符号名与函数名相同),并非改变函数的性质(也就是说被修饰的C++函数还可照常使用C++语言特性,与一般的C++函数除了多了个修饰之外,从外表上看并没有其它区别)。从如下示例程序中可以更清楚地看到这点:
// A.h
#ifndef MY_A_H #define MY_A_H
#include <iostream>
class A { public: void print() { std::cout<<"Message from class A"<<std::endl; } };
#endif //MY_A_H
////////////////////////////////////////////////////
// func.cpp
#include "A.h"
extern "C" void func(A& a) { int* p=new int[23]; // test new operator *p=1;
a.print();
delete [] p; }
//////////////////////////////////////////////// // main.cpp
#include "A.h"
extern "C" void func(A& a);
int main() { A a; func(a);
return 0; }
|
另外,标准中规定这种形式是非法的: extern "C" static void f(); // error 因为extern "C"表明它是外部连接的,与static矛盾(多存储类型声明错误)。但在VC6.0和MinGW2.05上这样声明却没有错误,不知道为什么会这样。不过,在定义时: extern "C" static void f(){...} // error 在VC6.0中没有编译错误,但会有连接错误(main中找不到func的定义);而MinGW2.05有编译错误(多个存储类型声明错误)。看来,MinGW工作得更好些。
使用extern "C"的一个很重要的因素是关于从动态库里提取函数等问题的需要,见下面的参考文献[2]。
参考文献: [1]ISO-IEC-14882-1998.pdf, 7.5 [2]关于从动态库里提取对象:http://www.isotton.com/howtos/C++-dlopen-mini-HOWTO/C++-dlopen-mini-HOWTO.html#seealso
|
|
|