符号修饰与函数签名、extern “C”(转载)
转自:http://www.cnblogs.com/monotone/archive/2012/11/16/2773772.html
参考资料:
- 《程序员的自我修养》3.5.3以及3.5.4小节。
符号修饰的由来
20世纪70年代以前,编译器编译代码时产生的目标文件中,符号名与相应的变量和函数的名字是一样的,随着编程语言的发展,例如C语言,如果一个C 语言程序要使用这些库的话,其自身就不能使用这些库中已经声明了的函数和变量的名字作为符号名,否则将会跟现有的目标文件发生名称冲突。
为了防止这类符号名冲突,各平台下的编程语言规定了各自的符号生成语法。如C在UNIX下在函数名和变量前加下划线作为符号名。这种给函数名增加特定符号来使其符号名唯一的方式就是符号修饰。
这种简单的符号修饰没有从根本上解决符号冲突的问题,比如同一种编程语言编写的目标文件之间还有可能产生符号冲突,当程序很大时,不同的部门之间也有可能会产生符号冲突,于是C++这类语言开始加上了命名空间(namespace)来解决多模块符号冲突问题。
为了支持C++拥有类、继承、虚机制、重载、命名空间等这些特性,人们发明了符号修饰修饰(或者称为符号改编),被修饰之前的函数名称以及其返回值和参数等信息,被称为函数签名。不同的编译器采用不同的名字修饰方法,因为导致由不同的编译器编译产生目标文件无法正常相互链接。
extern “C”
C++为了与C兼容,在符号的管理上,C++有一个用来声明或定一个C的符号的“extern ”C"”关键字用法:
- extern "C" int add(int a, int b);
- extern "C" int a;
- // 或者
- //extern "C"
- //{
- // int add(int a, int b);
- // int a;
- //};
注意:extern “C”中的C必须大写。
C++编译器会将在extern “C”的大括号内(或者后面的)代码当作C语言代码来处理。所以很明显,上面的代码就是为了将add函数和变量a声明成C的方式,因为有可能他们的定义是放在.c文件中的。
举例:
- // c_code.h file in c_project project
- #pragma once
- int add(int a, int b);
- // c_code.c file in c_project project
- #include "c_code.h"
- int add(int a, int b)
- {
- return a + b;
- }
如果我们有一个main.c,代码如下
- #include "c_code.h"
- #include "stdio.h"
- #include <conio.h>
- int main(void)
- {
- printf("%d + %d = %d\n", , , add(,));
- _getch();
- return ;
- }
编译链接运行都没错。那么现在我们把main.c文件重命名成main.cpp,代码不变,会发生链接错误: error LNK2019: unresolved external symbol "int __cdecl add(int,int)" (?add@@YAHHH@Z) referenced in function _main。这里的链接错误就是指没有找到符号“?add@@YAHHH@Z”。这个符号就是"int __cdecl add(int,int)"修饰后的符号名。
原因是C的符号修饰方式和C++的符号修饰方式不同,导致同样的声明会产生不同的符号名。这里main.cpp是一个cpp文件,会使用cpp的方式进行编译链接,而add的定义却是在.c文件里面,因而会链接不上。
接下来尝试使用extern “C”,让add函数在main.cpp中编译链接时,使用C的符号修饰方式,这里有两种方法:
- extern "C" int add(int a, int b); // 在这里声明add函数,不使用c_code.h头文件
- int main(void)
- {
- printf("%d + %d = %d\n", , , add(,));
- _getch();
- return ;
- }
- extern "C" // 显示的将c_code.h所有声明都使用C方式修饰
- {
- #include "c_code.h"
- }
- #include "stdio.h"
- #include <conio.h>
- int main(void)
- {
- printf("%d + %d = %d\n", , , add(,));
- _getch();
- return ;
- }
以上两种方式其实是一样的,这里因为c_code.h中只有一个函数,所以使用第一种方法也很简单,但是如果头文件中有很多函数声明,使用第二种方法就简单多了。
由于C++是对C语言的扩展,我们常常会需要使用C的库函数,这些库函数的定义都是用.c文件实现的,那么为了避免每次我们在使用库函数的时候,都去用extern “C”关键字修饰其头文件,这些标准库头文件中往往都包含了如下代码来解决这个问题。
- // c_code.h file in c_project project
- #pragma once
- #ifdef __cplusplus
- extern "C"{
- #endif
- int add(int a, int b);
- #ifdef __cplusplus
- }
- #endif
意思是,如果编译的时候发现__cplusplus宏已经定义,则给后面的函数声明都加上extern “C”,以用C方式修饰,否则不处理。而__cplusplus宏是C++编译器在编译C++程序时默认定义的宏(其实我们也可以自己在.cpp文件的顶 部定义一个宏),显然,在.cpp文件中包含这种头文件的时候,就会将这些个函数声明成用C方式修饰,而如果在.c文件中包含时,就不会加上extern “C”修饰。这就解释了为什么之前main.c能直接编译通过,而main.cpp不能的原因。
后记
其实这篇博文主要就为了介绍extern “C”的用法,在网上能搜到很多用法介绍,我也看了好几篇,但是就是感觉有种不是很透彻的感觉,于是就去找了这本参考书,看完之后就比较明了了。所以说,有些知识看起来很简单,但是如果不是很透彻理解的话,还是太容易忘记。
符号修饰与函数签名、extern “C”(转载)的更多相关文章
- c++11 符号修饰与函数签名、函数指针、匿名函数、仿函数、std::function与std::bind
一.符号修饰与函数签名 1.符号修饰 编译器将c++源代码编译成目标文件时,用函数签名的信息对函数名进行改编,形成修饰名.GCC的C++符号修饰方法如下: 1)所有符号都以_z开头 2)名字空间的名字 ...
- 转载----c++ static修饰的函数作用与意义
static修饰的函数叫做静态函数,静态函数有两种,根据其出现的地方来分类: 如果这个静态函数出现在类里,那么它是一个静态成员函数: 静态成员函数的作用在于:调用这个函数不会访问或者修改任何对象(非s ...
- DLL模块例2:使用__declspec(dllexport)导出函数,extern "C"规范修饰名称,隐式连接调用dll中函数
以下内容,我看了多篇文章,整合在一起,写的一个例子,关于dll工程的创建,请参考博客里另一篇文章:http://www.cnblogs.com/pingge/articles/3153571.html ...
- 【c++基础】static修饰的函数作用与意义
static修饰的函数作用与意义 static修饰的函数叫做静态函数,静态函数有两种,根据其出现的地方来分类: 如果这个静态函数出现在类里,那么它是一个静态成员函数: 静态成员函数的作用在于:调用这个 ...
- Python: 拷贝函数签名
使用场景有很多,比如C API在Python下很多都变成了(*args, **kwargs)的参数,这时候可能需要为其添加一个更严格签名来约束参数. 查了许多资料,能有效的拷贝函数签名貌似只能通过动态 ...
- const修饰虚函数
[1]程序1 #include <iostream> using namespace std; class Base { public: ; }; class Test : public ...
- 浅谈python函数签名
函数签名对象,表示调用函数的方式,即定义了函数的输入和输出. 在Python中,可以使用标准库inspect的一些方法或类,来操作或创建函数签名. 获取函数签名及参数 使用标准库的signature方 ...
- const 修饰成员函数 前后用法(effective c++ 03)
目录 const在函数后面 const修饰成员函数的两个作用 const在函数前面 总结 const在函数后面 类的成员函数后面加 const,表明这个函数不会对这个类对象的数据成员(准确地说是非静态 ...
- swift Equatable 函数签名的测试
struct Degoo:Equatable { var lex:String var pex:String static func == (left:Degoo, right:Degoo) -> ...
随机推荐
- Fiddler简介与Web抓包,远程抓包(IE、360、谷歌、火狐)
Fiddler简介以及web抓包 一.Fiddler简介 简单来说,Fiddler是一个http协议调试代理工具,它能够记录并检查所有你的电脑和互联网之间的http通讯.网上简介很多,我们不多说. 二 ...
- 九度oj 题目1063:整数和
题目1063:整数和 时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:4043 解决:2638 题目描述: 编写程序,读入一个整数N.若N为非负数,则计算N到2N之间的整数和:若N为一个负数 ...
- 郁闷的出纳员(bzoj 1503)
Description OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的工作,但是令人郁闷的是,我们的老板反复无常,经常 ...
- android源码编译时拷贝替换指定文件
由于要做版本定制,某些版本的资源文件等(例如style.xml)需要不同的配置,但是android的编译开关无法在xml里使用,于是想到了编译时根据不同的编译开关编译不同的文件,如下: 1.建立A.x ...
- 好用的window命令
Nslookup-------IP地址侦测器 chkdsk-----Chkdsk磁盘检查 regedt32-------注册表编辑器 regedit----注册表 perfmon----计算机性能监测 ...
- [JLOI2008]提示问题
题目描述 最近在JLOI网上的一个流行游戏中,选手要回答很难的问题.假如在规定时间内不能回答,系统将给出1个提示,之后再依次给出第2,3个提示.出现在答案中的是字母和下列字符: '.',',',':' ...
- msp430入门编程07
msp430中C语言的函数及实现07 msp430中C语言操作端口I/O10 msp430中C语言的模块化头文件及实现11 msp430中C语言的模块化头文件及库文件12 msp430入门学习 msp ...
- poj - 3041 Asteroids (二分图最大匹配+匈牙利算法)
http://poj.org/problem?id=3041 在n*n的网格中有K颗小行星,小行星i的位置是(Ri,Ci),现在有一个强有力的武器能够用一发光速将一整行或一整列的小行星轰为灰烬,想要利 ...
- POJ 3686_The Windy's
题意: N个工件要在M个工厂加工,一个工件必须在一个工厂做完,工厂一次只能处理一个工件.给定每个工件在每个工厂加工所需时间,求出每个工件加工结束的最小时间平均值. 分析: 工厂一次只能处理一个工件,那 ...
- Layui弹出层、日期和时间选择、即时通讯、分页
Layui弹出层.日期和时间选择.即时通讯.分页 弹层组件文档 - layui.layer 对于弹出层的感觉是:layer 至今仍作为 layui 的代表作,她的受众广泛并非偶然,而是这数年来的坚持. ...