extern "C"的作用

2023-05-16

extern "C"的作用(一) 



前些天,编程序是用到了很久以前写的C程序,想把里面的函数利用起来,连接发现出现了找不到具体函数的错误: 

以下是假设旧的C程序库 

C的头文件 

/*-----------c.h--------------*/ 
#ifndef _C_H_ 
#define _C_H_ 
extern int add(int x, int y); 
#endif 
C的源文件 

/*-----------c.c--------------*/ 
int add(int x, int y){ 
return x+y; 

C++的调用 

/*-----------cpp.cpp--------------*/ 
#include "c.h" 
void main() 

add(1, 0); 

这样编译会产生错误cpp.obj : error LNK2001: unresolved external symbol "int __cdecl add(int,int)" (?add@@YAHHH@Z),原因是找不到add的目标模块 

这才令我想起C++重载的函数命名方式和C函数的命名方式,让我们回顾一下:C中函数编译后命名会在函数名前加以"_",比如add函数编译成obj文件时的实际命名为_add,而c++命名则不同,为了实现函数重载同样的函数名add因参数的不同会被编译成不同的名字 

例如 

int add(int , int)==>add@@YAHHH@Z, 

float add(float , float )==>add@@YAMMM@Z, 

以上是VC6的命名方式,不同的编译器会不同,总之不同的参数同样的函数名将编译成不同目标名,以便于函数重载是调用具体的函数. 

编译cpp.cpp中编译器在cpp文件中发现add(1, 0);的调用而函数声明为extern int add(int x, int y);编译器就决定去找add@@YAHHH@Z,可惜他找不到,因为C的源文件把extern int add(int x, int y);编译成_add了; 

为了解决这个问题C++采用了extern "C",这就是我们的主题,想要利用以前的C程序库,那么你就要学会它,我们可以看以下标准头文件你会发现,很多头文件都有以下的结构 

#ifndef __H 
#define __H 
#ifdef __cplusplus 
extern "C" { 
#endif 

extern int f1(int, int); 
extern int f2(int, int); 
extern int f3(int, int); 


#ifdef __cplusplus 

#endif 

#endif /*__H*/ 

如果我们仿制该头文件可以得到 

#ifndef _C_H_ 
#define _C_H_ 
#ifdef __cplusplus 
extern "C" { 
#endif 

extern int add(int, int); 

#ifdef __cplusplus 

#endif 

#endif /* _C_H_ */ 

这样编译 

/*-----------c.c--------------*/ 
int add(int x, int y){ 
return x+y; 



这时源文件为*.c,__cplusplus没有被定义,extern "C" {}这时没有生效对于C他看到只是extern int add(int, int); 
add函数编译成_add(int, int); 

而编译c++源文件 

/*-----------cpp.cpp--------------*/ 
#include "c.h" 
void main() 

add(1, 0); 

这时源文件为*.cpp,__cplusplus被定义,对于C++他看到的是extern "C" {extern int add(int, int);}编译器就会知道 add(1, 0);调用的C风格的函数,就会知道去c.obj中找_add(int, int)而不是add@@YAHHH@Z; 

这也就为什么DLL中常看见extern "C" {},windows是采用C语言编制他首先要考虑到C可以正确调用这些DLL,而用户可能会使用C++而extern "C" {}就会发生作用 


extern "C"的作用(二) 



一、修饰名(Decorated Name) 
C/C++程序中的函数在内部是通过修饰名来标识的.修饰名是在函数定义或原型编译阶段由编译器创建字符串.当你在LINK等工具中要指定一个函数名时,会用到修饰名. 
1、使用修饰名: 
大多数情况下,你不必知道函数的修饰名是什么.连接器等工具通常都能处理函数未修饰的名字.然而,在有些情况下,你可能需要指定函数的修饰名.对于C++重载函数和特定的成员函数(如:构造函数和析构函数),你必须指定这些函数的修饰名,以便连接器等工具能够匹配名字.同时,你也必须在那些引用c或c++函数名的汇编源文件中使用修饰名. 
2、查看修饰名: 
如果你编译了一个源文件,该源文件中包含了函数定义或原型,你可以获得函数的修饰名形式. 
(1)用编译器列表(compiler listing)来查看: 
   (i)通过将列表文件类型编译器选项(/FA[c|s]) 设置为下面中的一种,来产生列表文件:Assembly with Machine Code (/FAc); Assembly with Source Code (/FAs); Assembly, Machine Code, and Source (/FAcs). 
   (ii)在产生的列表文件中,找到包含未经修饰的函数定义的行. 
   (iii)查找前面一行.PROC NEAR 命令标签前就是函数名经过修饰后的形式. 


(2)使用DUMPBIN工具来查看: 
  在.OBJ或.LIB上运行 DUMPBIN,使用/SYMBOLS选项.在输出中查找未经修饰的函数定义.后面跟着的就是经过修饰的函数名,用圆括号括起来的. 
二、替代连接说明: 
如果在c++中编写一个程序需要用到c的库,那该如何?如果这样声明一个c函数: 
void f(int a,char b); 
c++编译器就会将这个名字变成相应的修饰名,比如:?f@@YAXHD@Z. 
然而,c编译器编译的库的内部函数名(连接器使用)是完全不同的.这样,当c++连接器连接c的函数库时,将会产生内部使用函数不匹配. 
故,c++中提供了一个替代连接说明(alternate linkage specification),它是通过重载extern关键字来实现的. 
extern后跟一个字符串来指定想声明的函数的连接类型,后面是函数声明,比如: 
extern "C" void f(int a,char b); 
这样,就是告诉编译器是c连接,这样就不会转换函数名了.此例中,编译后的内部函数名是_f.
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

extern "C"的作用 的相关文章

随机推荐

  • C++QT5跨平台界面编程原理和实战大全-夏曹俊-专题视频课程

    C 43 43 QT5跨平台界面编程原理和实战大全 4271人已学习 课程介绍 课程讲解基于QT5 9 SDK xff0c 包含QT界面编程的核心内容 xff0c 并提供全部源码 xff0c 课程讲解方式是接口讲解 分析 演示示例 xff0
  • Darknet问题:ERROR: Cannot load message class for [darknet_ros_msgs/BoundingBoxes].

    运行darknet ros进行目标检测时 xff0c 通过 rostopic list查看到 darknet ros bounding boxes这个话题 xff0c 因此想查看该话题的数据 xff0c 在使用 rostopic echo
  • ROS问题:ERROR: cannot launch node of type

    在运行launch文件时 xff0c 出现如下错误提示 xff1a 原因是没有source 所以解决方案就是在编译完成之后添加source devel setup bash xff0c 问题解决 参考网址 xff1a ERROR canno
  • 标注工具labelme的使用

    在做目标检测任务时 xff0c 少不了对图像进行标注 xff0c 标注工具有好几种 xff0c labelme是其中比较好用的一种 labelme可对图像进行标注 xff0c 包括多边形 矩形 线 点和图像级标注 它是用Python编写的
  • Linux小技巧之终端terminal全选

    当打开一个终端 xff0c 经过若干指令后 xff0c 终端上输出的内容较多 xff0c 直接框选这些内容进行选择比较费事 有没有全选的功能呢 xff1f 答案是有的 xff01 方法1 xff1a 终端菜单栏全选 当窗口比较小时 xff0
  • 如何使用set::key_comp 和 set::value_comp 标准模板库 (STL) 函数

    下面的代码示例演示如何使用 Visual C 43 43 set key comp 和 set value comp 的 STL 功能 所需要的头文件 xff1a lt set gt 原型 template lt class K class
  • Linux小技巧之终端快捷键大全

    在前面一篇博客中记录了终端全选的技巧 下面记录一下关于终端使用的其它一些小技巧 F1查看帮助F11全屏Shift 43 Ctrl 43 T 打开一个新的终端Shift 43 Ctrl 43 N新建一个窗口打开终端Shift 43 Ctrl
  • ROS问题:Yolo v4移植到ROS后检测结果/darknet_ros/detection_image在rviz中显示乱码

    在前面一篇博客 xff08 Yolo v4移植ROS xff09 中介绍了将Yolo v4移植到ROS中 由于Yolo v4的源码在Yolo v3源码的基础上有改动 xff0c 移植成功后会出现一个小bug xff0c 如下图所示 xff1
  • 吉洪诺夫正则化(Tikhonov regularization )

    最近看了看吉洪诺夫正则化方法 xff0c 对其基本内容作了一个简单的了解 现在总结如下 1 正则化 定义 xff1a 正则化 regularization xff0c 是指在线性代数理论中 xff0c 不适定问题通常是由一组线性代数方程定义
  • C++中getline()、gets()等函数的用法

    在学习C 43 43 的过程中 xff0c 经常会遇到输入输出的问题 xff0c 以下总结一下下面几个函数的用法 xff1a 1 cin 2 cin get 3 cin getline 4 getline 5 gets 1 cin gt g
  • C++字母大小写转换方法

    字母大小写这个问题相对比较简单 xff0c 总结了一些常用的大小写转换的方法 xff0c 欢迎指正补充 xff01 思路1 xff1a 根据字母的ASCII表进行转换 xff1a 由表格可以看出 xff0c 对应大小写字母之间相差32 xf
  • C++ 标准输出控制小数点后位数的方法

    在C 43 43 中 xff0c 要实现这个功能 xff0c 就要用到std命名空间中常用于流的控制符 xff0c 这里通常要用到setprecision 函数 xff0c 可以通过这个函数控制小数点后面位数 还要注意的是 xff0c 使用
  • C++中string::npos的一些用法总结

    一 关于npos的定义 在MSDN中有如下说明 xff1a basic string npos static const size type npos 61 1 定义 The constant is the largest represen
  • CMake:通过target_link_libraries链接第三方库

    sdbusplus 通过new method call同步调用service的method 风静如云的博客 CSDN博客 例子中需要在编译时链接 lsdbusplus lsystemd 这两个第三方库 那么通过cmake怎么指定呢 其实很简
  • 在ubuntu终端打开谷歌浏览器的命令

    安装好谷歌浏览器后 xff0c 用以下命令在终端打开谷歌浏览器 adb shell am start n com android chrome com google android apps chrome Main 之后便出现如下内容 xf
  • PELCO_D通信协议

    1 球机通信接口 xff08 EIA RS 485 xff09 数据传输方式 xff1a 异步半双工串行通讯 通信波特率 xff1a 9600Bps 数据格式 xff1a Start Bit xff1a 1 Bit xff1b Data B
  • C buffer

    这学期在Dartmouth上ENGS20 Introduction to Scientific Computing xff0c 好多东西不记下来就会忘 xff0c 所以开一个笔记 在C语言中 xff0c 输入和输出都是有buffer的 xf
  • 寄存器值的操作方法

    通过这段时间的工作和学习 xff0c 我感觉在嵌入式硬件编程中 xff0c 大多数情况下都是对相应硬件的功能寄存器进行设置和操作 一 寄存器的设置和操作特性 1 xff0c 一个寄存器的每个位有其不同的意义 xff0c 进行不同的设置会使硬
  • UART串口通信(回环测试)

    一 UART串口通信简介 UART xff08 Universal Asynchronous Receiver Transmitter xff09 是采用异步串行通信方式的通用异步收发传输器 xff0c 在发送数据时将并行数据转换为串行数据
  • extern "C"的作用

    extern 34 C 34 的作用 一 前些天 编程序是用到了很久以前写的C程序 想把里面的函数利用起来 连接发现出现了找不到具体函数的错误 以下是假设旧的C程序库 C的头文件 c h ifndef C H define C H exte