符号修饰与函数签名、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) -> ...
随机推荐
- 集训第五周动态规划 G题 回文串
Description A palindrome is a symmetrical string, that is, a string read identically from left to ri ...
- JQuery_九大选择器
JQuery_九大选择器-----https://blog.csdn.net/pseudonym_/article/details/76093261
- 九度oj 题目1181:遍历链表
题目1181:遍历链表 时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:3483 解决:1465 题目描述: 建立一个升序链表并遍历输出. 输入: 输入的每个案例中第一行包括1个整数:n(1 ...
- Linux笔记:定时任务和文件操作
查看定时任务 crontab -l 注册定时任务 crontab -e然后就像 vim 一样编辑自己的定时任务.如: * * * * * . /home/hadoop/timer/check_job. ...
- Linux下汇编语言学习笔记30 ---
这是17年暑假学习Linux汇编语言的笔记记录,参考书目为清华大学出版社 Jeff Duntemann著 梁晓辉译<汇编语言基于Linux环境>的书,喜欢看原版书的同学可以看<Ass ...
- AOJ 0118 Property Distribution (DFS)
http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=46522 简单DFS,题目翻译参考 http://blog.csdn.net ...
- TCP/IP学习笔记(5)------IP选路
静态IP选路 一个简单的路由表 选路是IP层最重要的一个功能之一.前面的部分已经简单的讲过路由器是通过何种规则来根据IP数据包的IP地址来选择路由.这里就不重复了.首先来看看一个简单的系统路由表. D ...
- cogs——2084. Asm.Def的基本算法
2084. Asm.Def的基本算法 ★☆ 输入文件:asm_algo.in 输出文件:asm_algo.out 简单对比时间限制:1 s 内存限制:256 MB [题目描述] “有句 ...
- 源码SDWebImage
源码来源:https://github.com/rs/SDWebImage 版本: 3.7 SDWebImage是一个开源的第三方库,它提供了UIImageView的一个分类,以支持从远程服务器下载并 ...
- 手动安装Firefox Linux
(2015-06-05 17:22:19)[编辑][删除] 转载▼ 标签: 股票 Firefox 下载文件以.tar和.bz2格式保存,必须从这些压缩包中提取文件.不想删除当前安装的 Firefox, ...