名字修饰约定extern "C"与extern "C++"浅析

 所谓名字修饰约定,就是指变量名、函数名等经过编译后重新输出名称的规则。

  比如源代码中函数名称为int Func(int a,int b),经过编译后名称可能为[email protected]@[email protected]、[email protected]@[email protected]、[email protected],也有可能与源代码中名称相同为Func。

  影响编译后输出的名称通常与名字修饰约定(extern "C"、extern "C++"等)和函数调用约定(__stdcall、__cdecl等)等相关。

  口说千遍,不如实际演练一遍。那么,就让我们写代码来测试下。

  注意,本文只讨论extern "C"、extern "C++"和__stdcall、__cdecl相关的约定,其他约定不在本文讨论范围内。另外,编译的环境为XP + VC++6.0SP6。

  首先,用C方式导出两个函数:

  Dll1.c

[cpp] view plain copy

  1. _declspec(dllexport) int __cdecl Func_cdecl(int a,int b)
  2. {
  3. return 1;
  4. }
  5. _declspec(dllexport) int __stdcall Func_stdcall(int a,int b)
  6. {
  7. return 1;
  8. }

  导出的两个函数名为:

  再以C++方式导出:

  Dll1.cpp

[cpp] view plain copy

  1. _declspec(dllexport) int __cdecl Func_cdecl(int a,int b)
  2. {
  3. return 1;
  4. }
  5. _declspec(dllexport) int __stdcall Func_stdcall(int a,int b)
  6. {
  7. return 1;
  8. }

  导出结果如下:

  然后,我们再以C++方式导出如下代码中的函数:

[cpp] view plain copy

  1. extern "C" _declspec(dllexport) int __stdcall Func_C_stdcall(int a,int b)
  2. {
  3. return 1;
  4. }
  5. extern "C++" _declspec(dllexport) int __stdcall Func_CPP_stdcall(int a,int b)
  6. {
  7. return 1;
  8. }
  9. extern "C" _declspec(dllexport) int __cdecl Func_C_cdecl(int a,int b)
  10. {
  11. return 1;
  12. }
  13. extern "C++" _declspec(dllexport) int __cdecl Func_CPP_cdecl(int a,int b)
  14. {
  15. return 1;
  16. }

  导出结果如下:

  有了以上实验结果,我们再结合以下名字输出规则进行理解:

  1. C方式编译(extern "C"):

    1. __stdcall调用约定:输出名称在原名称前加一下划线,后面再加上一个“@”和其参数的总字节数(_原名称@参数总字节数),如名称int Func_C_stdcall(int a,int b)输出为[email protected];
    2. __cdecl调用约定:与原名称相同,如名称int Func_C_cdecl(int a,int b)输出还是为Func_C_cdecl;
  2. C++方式编译(extern "C++"):
    1. __stdcall调用约定:

      1. 输出名称以“?”开始,后跟原名称;
      2. 原名称后再跟“@@YG”,后面再跟返回值代号和参数表代号,代号表示如下:
        X--void ,
        D--char,
        E--unsigned char,
        F--short,
        H--int,
        I--unsigned int,
        J--long,
        K--unsigned long,
        M--float,
        N--double,
        _N--bool,
        ...
        PA--表示指针,后面的代号表明指针类型,如果相同类型的指针连续出现,以“0”代替,一个“0”代表一次重复;
      3. 参数表后以“@Z”标识整个名字的结束,如果该函数无参数,则以“Z”标识结束。如名称int Func_CPP_stdcall(int a,int b)编译后的输出名称为[email protected]@[email protected]。
    2. __cdecl调用约定:与_stdcall调用约定基本一致,只是参数表的开始标识由上面的“@@YG”变为“@@YA”。如名称int Func_CPP_cdecl(int a,int b)编译后输出名称为[email protected]@[email protected]。

  

  有个这个规则,再回头去看我们的实验结果,就很好理解了。

  当然,编译C文件和编译CPP文件,不需加extern "C"和extern "C++",因为编译C文件当然默认的是extern "C",而编译CPP文件则默认的是extern "C++"。

  现在我们也能理解为什么导出DLL时通常需要加上extern "C"。试想,如果一个C++导出的dll,没有加extern "C",则导出的名称为extern "C++"约定下的名称。如果这个dll需要提供给用C编写的程序使用,那么这个程序是无法调用这个dll的,因为C写的程序遵循的是extern "C"约定,链接时链接器将按照extern "C"约定的名称去寻找外部名称,这当然找不到,因为dll中的输出名称为extern "C++"约定下的名称。

时间: 2024-10-11 19:55:33

名字修饰约定extern "C"与extern "C++"浅析的相关文章

C++编译时函数名修饰约定规则(很具体),MFC提供的宏,extern "C"的作用

调用约定: __cdecl __fastcall与 __stdcall,三者都是调用约定(Calling convention),它决定以下内容:1)函数参数的压栈顺序,2)由调用者还是被调用者把参数弹出栈,3)以及产生函数修饰名的方法. 1.__stdcall调用约定:函数的参数自右向左通过栈传递,被调用的函数在返回前清理传送参数的内存栈, 2._cdecl是C和C++程序的缺省调用方式.每一个调用它的函数都包含清空堆栈的代码,所以产生的可执行文件大小会比调用_stdcall函数的大.函数采用

[转][C/C++]函数名字修饰(Decorated Name)方式

函数的名字修饰(Decorated Name)就是编译器在编译期间创建的一个字符串,用来指明函数的定义或原型.LINK程序或其他工具有时需要指定函数的名字修饰来定位函数的正确位置.多数情况下程序员并不需要知道函数的名字修饰,LINK程序或其他工具会自动区分他们.当然,在某些情况下需要指定函数的名字修饰,例如在C++程序中,为了让LINK程序或其他工具能够匹配到正确的函数名字,就必须为重载函数和一些特殊的函数(如构造函数和析构函数)指定名字装饰.另一种需要指定函数的名字修饰的情况是在汇编程序中调用

c++函数调用约定学习(一)

函数调用约定 常见的函数调用约定[5]:cdecl,stdcall,fastcall,thiscall,naked call MFC调用约定(VS6:Project Settings->C/C++ <Category:Code Generation> Calling convention:) 1, __cdecl(C调用约定.The C default calling convention)C/C++ 缺省调用方式 1)压栈顺序:函数参数从右到左 2)参数栈维护:由调用函数把参数弹出栈,

__stdcall修饰符

被这个关键字修饰的函数,其参数都是从右向左通过堆栈传递的(__fastcall 的前面部分由ecx,edx传). 函数调用在返回前要由被调用者清理堆栈. 扩展: 1.修饰名(Decoration name) “C”或者“C++”函数在内部(编译和链接)通过修饰名识别. 修饰名是编译器在编译函数定义或者原型时生成的字符串. 有些情况下使用函数的修饰名是必要的,如在模块定义文件里头指定输出“C++”重载函数.构造函数.析构函数,又如在汇编代码里调用“C””或“C++”函数等. 修饰名由函数名.类名.

__stdcall,__cdecl,__fastcall的区别

__stdcall,__cdecl,__fastcall的区别 标签: dll编译器pascalclassimportinitialization 2009-12-09 15:07 10472人阅读 评论(1) 收藏 举报 分类: C/C++(22) __stdcall,__cdecl,__fastcall的区别 一.三者区别一览表 __stdcall __cdecl __fastcall 参数传递方式 右->左 压栈 右->左 压栈 左边开始的两个不大于4字节(DWORD)的参数分别放在EC

函数的调用规则(__cdecl,__stdcall,__fastcall,__pascal) http://blog.csdn.net/kaiwii/article/details/8500686

函数的调用规则(__cdecl,__stdcall,__fastcall,__pascal) 2013-01-14 13:51 1548人阅读 评论(0) 收藏 举报  分类: android底层(14)  c&c++(18)  Linux下如何指定调用约定(calling convention) Windows下的调用约定可以是stdcall/cdecl/fastcall,这些标识加在函数名前面,如: int __stdcall funca() 但在Linux下,如按照上面写法后,编译程序将导

_stdcall ,_cdecl,__fastcall 深入解析

成相对独立的功能,它们彼此协作来完成整个软件系统的工作.可能存在一些模块的功能较为通用,在构造其它软件系统时仍会被使用.在构造软件系统时,如果将所有模块的源代码都静态编译到整个应用程序 EXE 文件中,会产生一些问题:一个缺点是增加了应用程序的大小,它会占用更多的磁盘空间,程序运行时也会消耗较大的内存空间,造成系统资源的浪费:另一个缺点是,在编写大的 EXE 程序时,在每次修改重建时都必须调整编译所有源代码,增加了编译过程的复杂性,也不利于阶段性的单元测试. Windows 系统平台上提供了一种

【转】分析Linux和windows动态库

原文地址:http://www.cnblogs.com/chio/archive/2008/11/13/1333119.html 摘要:动态链接库技术实现和设计程序常用的技术,在Windows和Linux系 统中都有动态库的概念,采用动态库可以有效的减少程序大小,节省空间,提高效率,增加程序的可扩展性,便于模块化管理.但不同操作系统的动态库由 于格式不同,在需要不同操作系统调用时需要进行动态库程序移植.本文分析和比较了两种操作系统动态库技术,并给出了将Visual C++编制的动态库移植到Lin

Linux和windows动态库

转载:http://www.cnblogs.com/chio/archive/2008/11/13/1333119.html 态链接库技术实现和设计程序常用的技术,在Windows和Linux系 统中都有动态库的概念,采用动态库可以有效的减少程序大小,节省空间,提高效率,增加程序的可扩展性,便于模块化管理.但不同操作系统的动态库由 于格式不同,在需要不同操作系统调用时需要进行动态库程序移植.本文分析和比较了两种操作系统动态库技术,并给出了将Visual C++编制的动态库移植到Linux上的方法