`
rbt359um
  • 浏览: 11569 次
最近访客 更多访客>>
社区版块
存档分类
最新评论

探索C++的秘密之详解extern C

 
阅读更多

探索C++的秘密之详解extern C
2010年05月21日
  时常在cpp的代码之中看到这样的代码: PnQ
  PnQ
  #ifdef __cplusplus PnQ
  extern "C" { PnQ
  #endif PnQ
  PnQ
  //一段代码 PnQ
  PnQ
  #ifdef __cplusplus PnQ
  } PnQ
  #endif PnQ
    这样的代码到底是什么意思呢?首先,__cplusplus是cpp中的自定义宏,那么定义了这个宏的话表示这是一段cpp的代码,也就是说,上面的代码的含义是:如果这是一段cpp的代码,那么加入extern "C"{和}处理其中的代码。 PnQ
  PnQ
    要明白为何使用extern "C",还得从cpp中对函数的重载处理开始说起。在c++中,为了支持重载机制,在编译生成的汇编码中,要对函数的名字进行一些处理,加入比如函数的返回类型等等.而在C中,只是简单的函数名字而已,不会加入其他的信息.也就是说:C++和C对产生的函数名字的处理是不一样的. PnQ
  PnQ
    比如下面的一段简单的函数,我们看看加入和不加入extern "C"产生的汇编代码都有哪些变化: PnQ
  PnQ
  int f(void) PnQ
  { PnQ
  return 1; PnQ
  } PnQ
    在加入extern "C"的时候产生的汇编代码是: PnQ
  PnQ
  .file "test.cxx" PnQ
  .text PnQ
  .align 2 PnQ
  .globl _f PnQ
  .def _f; .scl 2; .type 32; .endef PnQ
  _f: PnQ
  pushl %ebp PnQ
  movl %esp, %ebp PnQ
  movl $1, %eax PnQ
  popl %ebp PnQ
  ret PnQ
    但是不加入了extern "C"之后 PnQ
  PnQ
  .file "test.cxx" PnQ
  .text PnQ
  .align 2 PnQ
  .globl __Z1fv PnQ
  .def __Z1fv; .scl 2; .type 32; .endef PnQ
  __Z1fv: PnQ
  pushl %ebp PnQ
  movl %esp, %ebp PnQ
  movl $1, %eax PnQ
  popl %ebp PnQ
  ret PnQ
    两段汇编代码同样都是使用gcc -S命令产生的,所有的地方都是一样的,唯独是产生的函数名,一个是_f,一个是__Z1fv。 PnQ
  PnQ
    明白了加入与不加入extern "C"之后对函数名称产生的影响,我们继续我们的讨论:为什么需要使用extern "C"呢?C++之父在设计C++之时,考虑到当时已经存在了大量的C代码,为了支持原来的C代码和已经写好C库,需要在C++中尽可能的支持C,而extern "C"就是其中的一个策略。 PnQ
  PnQ
    试想这样的情况:一个库文件已经用C写好了而且运行得很良好,这个时候我们需要使用这个库文件,但是我们需要使用C++来写这个新的代码。如果这个代码使用的是C++的方式链接这个C库文件的话,那么就会出现链接错误.我们来看一段代码:首先,我们使用C的处理方式来写一个函数,也就是说假设这个函数当时是用C写成的: PnQ
  PnQ
  //f1.c PnQ
  extern "C" PnQ
  { PnQ
  void f1() PnQ
  { PnQ
  return; PnQ
  } PnQ
  } PnQ
    编译命令是:gcc -c f1.c -o f1.o 产生了一个叫f1.o的库文件。再写一段代码调用这个f1函数: PnQ
  PnQ
  // test.cxx PnQ
  //这个extern表示f1函数在别的地方定义,这样可以通过 PnQ
  //编译,但是链接的时候还是需要 PnQ
  //链接上原来的库文件. PnQ
  extern void f1(); PnQ
  PnQ
  int main() PnQ
  { PnQ
  f1(); PnQ
  PnQ
  return 0; PnQ
  } PnQ
    通过gcc -c test.cxx -o test.o 产生一个叫test.o的文件。然后,我们使用gcc test.o f1.o来链接两个文件,可是出错了,错误的提示是: PnQ
  PnQ
  test.o(.text + 0x1f):test.cxx: undefine reference to 'f1()' PnQ
    也就是说,在编译test.cxx的时候编译器是使用C++的方式来处理f1()函数的,但是实际上链接的库文件却是用C的方式来处理函数的,所以就会出现链接过不去的错误:因为链接器找不到函数。 PnQ
  PnQ
    因此,为了在C++代码中调用用C写成的库文件,就需要用extern "C"来告诉编译器:这是一个用C写成的库文件,请用C的方式来链接它们。 PnQ
  PnQ
    比如,现在我们有了一个C库文件,它的头文件是f.h,产生的lib文件是f.lib,那么我们如果要在C++中使用这个库文件,我们需要这样写: PnQ
  PnQ
  extern "C" PnQ
  { PnQ
  #include "f.h" PnQ
  } PnQ
    回到上面的问题,如果要改正链接错误,我们需要这样子改写test.cxx: PnQ
  PnQ
  extern "C" PnQ
  { PnQ
  extern void f1(); PnQ
  } PnQ
  PnQ
  int main() PnQ
  { PnQ
  f1(); PnQ
  PnQ
  return 0; PnQ
  } PnQ
    重新编译并且链接就可以过去了. PnQ
  PnQ
    总结
PnQ
  PnQ
    C和C++对函数的处理方式是不同的.extern "C"是使C++能够调用C写作的库文件的一个手段,如果要对编译器提示使用C的方式来处理函数的话,那么就要使用extern "C"来说明。
分享到:
评论

相关推荐

    探索C++的秘密之详解extern

    本文主要介绍了C+中extern的用法,希望对你的学习有所帮助。

    C++中的extern “C”用法详解

    主要介绍了C++中的extern “C”用法详解,简单来说,extern “C”是C++声明或定义C语言符号的方法,是为了与C兼容,需要的朋友可以参考下

    extern用法详解

    extern用法详解 讲述 c\c++ extern关键字的用法…… 貌似非入门必备,但有备无患……

    C++调用C函数实例详解

    C++调用C函数实例详解 前言:以前见到extern “C”这样的语句,只是简单地知道跟外部链接有关,但是没有深刻理解它的意思。 首先,为什么要使用extern “C”修饰符? C++调用其它语言的函数,由于编译器生成函数的...

    C++中的extern声明变量详解

    今天我们只谈extern,什么const、static之类等等与之相关或不相关的一律忽略,下面就分别对以上两种情况一一讲解 声明和定义 既然提到extern声明变量,那我们就必须搞清楚声明和定义的区别。 这里我们将普通数据变量...

    浅谈C/C++中的static与extern关键字的使用详解

    本篇文章是对C/C++中的static与extern关键字的使用进行了详细的分析介绍,需要的朋友参考下

    C/C++ 中extern关键字详解

    C/C++ 中extern关键字详解 在C/C++编程过程中,经常会进行变量和函数的声明和定义,各个模块间共用同一个全局变量时,此时extern就派上用场了。 定义 extern可以置于变量或者函数前,以标示变量或者函数的定义在别的...

    实例详解C/C++中extern关键字

    主要介绍了C/C++中extern关键字详解 的相关资料,需要的朋友可以参考下

    C/C++中extern “C” 的作用分析

    我们经常会在C/C++程序中见到extern “C”,这是一个很重要的概念。本文就来以实例形式讲述C/C++中extern “C”的作用。分享给大家供大家参考之用。具体分析如下: 作用:实现C和C++混合编程。 原理:C和C++编译器...

    C/C++中extern关键字详解

    则告诉编译器在编译fun这个函数名时按着C的规则去翻译相应的函数名而不是C++的,C++的规则在翻译这个函数名时会把fun这个名字变得面目全非,可能是fun@aBc_int_int#%$也可能是别的,这要看编译器的脾气了(不同的...

    高级C语言详解

    60. 高级C语言程序员测试必过的十六道最佳题目+答案详解 297 61. C语言常见错误 320 62. 超强的指针学习笔记 325 63. 程序员之路──关于代码风格 343 64. 指针、结构体、联合体的安全规范 346 65. C指针讲解 352 66...

    详解python如何调用C/C++底层库与互相传值

    1.extern “C” 导出(互相传值比较麻烦,不建议使用这种方式): 将C/C++库做成和平常一样的DLL和或者.so,比如: //.h文件 #include //.cpp文件 //C/C++ my.so 或者my.dll enter "C" void printHello() { std:...

    C++100天经典实战系(含大量案例源码及通俗讲解教程,effective_c++、STL、综合能力全面提升)

    class、struct、static、sizeof、inline那些事、func_pointer、extern C、explicit(显式)关键字那些事、decltype那些事、const介绍及案例、C++面向对象特性 2、10日c++实战狂练(习题和源码) 3、C++11新特性详解 4...

    详解C++中的const关键字及与C语言中const的区别

    要是const变量能在其他文件中访问,必须显示的指定extern(c中也是)   当你只在定义该const常量的文件中使用该常量时,c++不给你的const常量分配空间–这也是c++的一种优化措施,没有必要浪费内存空间来存储一个...

Global site tag (gtag.js) - Google Analytics