转 欢迎使用CSDN-markdown编辑器

2023-10-31

https://blog.csdn.net/CSDNwei/article/details/51836182
Extern的问题在于不知道这个关键词出现的时候到底是声明还是定义。

谨记:声明可以多次,定义只能一次。

函数的声明extern关键词是可有可无的,因为函数本身不加修饰的话就是extern的。但是引用的时候一样是需要声明的。

而全局变量在外部使用声明时,extern关键词是必须的,如果变量无extern修饰且没有显式的初始化,同样成为变量的定义,因此此时必须加extern,而编译器在此标记存储空间在执行时加载如内存并初始化为0。

而局部变量的声明不能有extern的修饰,且局部变量在运行时才在堆栈部分分配内存。

引用性声明、定义性声明

强符号、弱符号

出现在linux的gcc链接分析中,可以加深链接的理解。

全局变量或函数本质上讲没有区别,函数名是指向函数二进制块开头处的指针。而全局变量是在函数外部声明的变量。函数名也在函数外,因此函数也是全局的。

在使用中,要形成一种风格。

头文件

首先说下头文件,其实头文件对计算机而言没什么作用,她只是在预编译时在#include的地方展开一下,没别的意义了,其实头文件主要是给别人看的。

我做过一个实验,将头文件的后缀改成xxx.txt,然后在引用该头文件的地方用

#include"xxx.txt"

编译,链接都很顺利的过去了,由此可知,头文件仅仅为阅读代码作用,没其他的作用了!

不管是C还是C++,你把你的函数,变量或者结构体,类啥的放在你的.c或者.cpp文件里。然后编译成lib,dll,obj,.o等等,然后别人用的时候最基本的gcc hisfile.cpp yourfile.o|obj|dll|lib 等等。
但对于我们程序员而言,他们怎么知道你的lib,dll…里面到底有什么东西?要看你的头文件。你的头文件就是对用户的说明。函数,参数,各种各样的接口的说明。
那既然是说明,那么头文件里面放的自然就是关于函数,变量,类的“声明”了。记着,是“声明”,不是“定义”。
那么,我假设大家知道声明和定义的区别。所以,最好不要傻嘻嘻的在头文件里定义什么东西。比如全局变量:

#ifndef _XX_头文件.H
#define _XX_头文件.H
int A;
#endif

那么,很糟糕的是,这里的int A是个全局变量的定义,所以如果这个头文件被多次引用的话,你的A会被重复定义
显然语法上错了。只不过有了这个#ifndef的条件编译,所以能保证你的头文件只被引用一次,不过也许还是会岔子,但若多个c文件包含这个头文件时还是会出错的,因为宏名有效范围仅限于本c源文件,所以在这多个c文件编译时是不会出错的,但在链接时就会报错,说你多处定义了同一个变量,

Linking…
incl2.obj : error LNK2005: “int glb” (?glb@@3HA) already defined in incl1.obj
Debug/incl.exe : fatal error LNK1169: one or more multiply defined symbols found

注意!!!

extern

这个关键字真的比较可恶,在声明的时候,这个extern居然可以被省略,所以会让你搞不清楚到底是声明还是定义,下面分变量和函数两类来说:

(1)变量

尤其是对于变量来说。
extern int a;//声明一个全局变量a
int a; //定义一个全局变量a

extern int a =0 ;//定义一个全局变量a 并给初值。
int a =0;//定义一个全局变量a,并给初值,

第四个 等于 第 三个,都是定义一个可以被外部使用的全局变量,并给初值。
糊涂了吧,他们看上去可真像。但是定义只能出现在一处。也就是说,不管是int a;还是extern int a=0;还是int a=0;都只能出现一次,而那个extern int a可以出现很多次。

当你要引用一个全局变量的时候,你就要声明,extern int a;这时候extern不能省略,因为省略了,就变成int a;这是一个定义,不是声明。

(2)函数
函数,函数,对于函数也一样,也是定义和声明,定义的时候用extern,说明这个函数是可以被外部引用的,声明的时候用extern说明这是一个声明。 但由于函数的定义和声明是有区别的,定义函数要有函数体,声明函数没有函数体,所以函数定义和声明时都可以将extern省略掉,反正其他文件也是知道这个函数是在其他地方定义的,所以不加extern也行。两者如此不同,所以省略了extern也不会有问题。
比如:

int fun(void)
{
return 0;
}

很好,我们定义了一个全局函数

int fun(void);
我们对它做了个声明,然后后面就可以用了
加不加extern都一样
我们也可以把对fun的声明 放在一个头文件里,最后变成这样

int fun(void);//函数声明,所以省略了extern,完整些是extern int fun(void);

int fun(void)
{
return 0;
}//一个完整的全局函数定义,因为有函数体,extern同样被省略了。
然后,一个客户,一个要使用你的fun的客户,把这个头文件包含进去,ok,一个全局的声明。没有问题。
但是,对应的,如果是这个客户要使用全局变量,那么要extern 某某变量;不然就成了定义了。

总结下:

对变量而言,变量的声明有两种情况: 一种是需要建立存储空间的,不用加extern;2、另一种是不需要建立存储空间,需要加extern 。如果你想在本源文件中使用另一个源文件的变量,就需要在使用前用extern声明该变量,或者在头文件中用extern声明该变量;

对函数而言,如果你想在本源文件中使用另一个源文件的函数,就需要在使用前用声明该函数,声明函数加不加extern都没关系,所以在头文件中函数可以不用加extern。

extern "C"的用法
链接指示符extern C
如果程序员希望调用其他程序设计语言尤其是C 写的函数,那么调用函数时必须告诉编译器使用不同的要求,例如当这样的函数被调用时函数名或参数排列的顺序可能
不同,无论是C++函数调用它还是用其他语言写的函数调用它,程序员用链接指示符linkage directive 告诉编译器该函数是用其他的程序设计语言编写的,链接指示符有两种形式既可以是单一语句single statement 形式也可以是复合语句compound statement 形式。
// 单一语句形式的链接指示符
extern “C” void exit(int);
// 复合语句形式的链接指示符
extern “C” {
int printf( const char* … );
int scanf( const char* … );
}
// 复合语句形式的链接指示符
extern “C” {
#include
}
链接指示符的第一种形式由关键字extern 后跟一个字符串常量以及一个普通的函数,声明构成虽然函数是用另外一种语言编写的但调用它仍然需要类型检查例如编译器会检查传递给函数exit()的实参的类型是否是int 或者能够隐式地转换成int 型,多个函数声明可以用花括号包含在链接指示符复合语句中,这是链接指示符的第二种形式花扩号被用作分割符表示链接指示符应用在哪些声明上在其他意义上该花括号被忽略,所以在花括号中声明的函数名对外是可见的就好像函数是在复合语句外声明的一样,例如在前面的例子中复合语句extern "C"表示函数printf()和scanf()是在C 语言中写的,函数因此这个声明的意义就如同printf()和scanf()是在extern "C"复合语句外面声明的一样,当复合语句链接指示符的括号中含有#include 时在头文件中的函数声明都被假定是用链接指示符的程序设计语言所写的在前面的例子中在头文件中声明的函数都是C函数链接指示符不能出现在函数体中下列代码段将会导致编译错误。
int main()
{
// 错误: 链接指示符不能出现在函数内
extern “C” double sqrt( double );
305 第七章函数
double getValue(); //ok
double result = sqrt ( getValue() );
//…
return 0;
}
如果把链接指示符移到函数体外程序编译将无错误
extern “C” double sqrt( double );
int main()
{
double getValue(); //ok
double result = sqrt ( getValue() );
//…
return 0;
}
但是把链接指示符放在头文件中更合适在那里函数声明描述了函数的接口所属,如果我们希望C++函数能够为C 程序所用又该怎么办呢我们也可以使用extern "C"链接指示符来使C++函数为C 程序可用例如。
// 函数calc() 可以被C 程序调用
extern “C” double calc( double dparm ) { /* … */ }
如果一个函数在同一文件中不只被声明一次则链接指示符可以出现在每个声明中它,也可以只出现在函数的第一次声明中在这种情况下第二个及以后的声明都接受第一个声
明中链接指示符指定的链接规则例如
// ---- myMath.h ----
extern “C” double calc( double );
// ---- myMath.C ----
// 在Math.h 中的calc() 的声明
#include “myMath.h”
// 定义了extern “C” calc() 函数
// calc() 可以从C 程序中被调用
double calc( double dparm ) { // …
在本节中我们只看到为C 语言提供的链接指示extern “C”,extern "C"是惟一被保证由所有C++实现都支持的,每个编译器实现都可以为其环境下常用的语言提供其他链接指示例如extern "Ada"可以用来声明是用Ada 语言写的函数,extern "FORTRAN"用来声明是用FORTRAN 语言写的函数,等等因为其他的链接指示随着具体实现的不同而不同所以建议读者查看编译器的用户指南以获得其他链接指示符的进一步信息。

总结 extern “C”

   extern “C” 不但具有传统的声明外部变量的功能,还具有告知C++链接器使用C函数规范来链接的功能。 还具有告知C++编译器使用C规范来命名的功能。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

转 欢迎使用CSDN-markdown编辑器 的相关文章

随机推荐

  • Android Studio : cmdline-tools component is missing. Android license status unknown.

    运行flutter doctor报错 D AndroidStudio projectLocations gt flutter doctor Doctor summary to see all details run flutter doct
  • Caused by: java.lang.IllegalStateException: Logback configuration error detected:

    项目场景 今天像往常一样打了个jar包 在本地是可以运行的 放测试环境上就不能运行了 整了半小时才看明白 以后一定要认识看报错信息 问题描述 在本地测试可以运行 在测试环境报 Caused by java lang IllegalState
  • 【自学C++】C++ short

    C short C short教程 C 中的 short 用来表示一个 整数 也可以叫做短整型 如果我们需要表示的整数比较小 那么我们可以使用 short 来定义 这样可以节省系统资源 C short定义详解 语法 short int va
  • python学习-面向对象进阶之闭包(八)

    闭包 1 函数引用 def test1 print in test1 func 调用函数 test1 引用函数 ret test1 print id ret print id test1 通过引用调用函数 ret 运行结果 in test1
  • 第19章 通信-模拟I2C

    一 关于I2C 1 1 I2C协议 I C Inter Integrated Circuit 常读作 I方C 它是一种多主从架构串行通信总线 在1980年由飞利浦公司设计 用于让主板 嵌入式系统或手机连接低速周边设备 I C由两条线组成 一
  • JAVA开发(外部接口调用授权问题记录总结)

    一 技术背景 现在很多web项目或者小程序在上线后都需要进行交叉引流 交叉业务合作 数据传输 与其他的企业 网站 app合作 那么就需要接口数据调用 那么在做外部系统接口调用和自己开发的微服务间的接口调用显然是不同的 最明显的特征就是 系统
  • Simple_SSTI_1与Simple_SSTI_2

    目录 一 Simple SSTI 1 二 Simple SSTI 2 一 Simple SSTI 1 首先打开场景 然后F12查看一下源码 于是通过百度相关知识寻找线索 1 SSTI 服务器端模版注入是指攻击者能够使用本机模板语法将恶意有效
  • BES2500/BES2300 学习笔记目录

    交流学习 如有错误欢迎指正 by wwdeng QQ 435398366 BES2500 BES2300 学习笔记目录 开发环境 代码编辑器 vscode Source Insight 编译 Windows Linux 调试 Trace 工
  • HTML5知识点总结

    1 什么是 HTML5 1 1 HTML版本 1 2 HTML5简介 万维网的核心语言 标准通用标记语言下的一个应用超文本标记语言 HTML 的第五次重大修改 用于取代HTML4与 XHTML的新一代标准版本 所以叫HTML5 XHTML
  • Spring-Resource接口

    4 1 1 概述 在日常程序开发中 处理外部资源是很繁琐的事情 我们可能需要处理URL资源 File资源资源 ClassPath相关资源 服务器相关资源 JBoss AS 5 x上的VFS资源 等等很多资源 因此处理这些资源需要使用不同的接
  • 【云原生之kubernetes】k8s集群的日常基本操作

    云原生之kubernetes k8s集群的日常基本操作 一 查看集群的状态 1 1 查看集群的节点 1 2 查看节点的详细状态 1 3 查看当前集群的pod 1 4 查看pod的详细信息 1 5 查看集群的所有pods 1 6 查看depo
  • 从贝叶斯方法谈到贝叶斯网络

    从贝叶斯方法谈到贝叶斯网络 0 引言 事实上 介绍贝叶斯定理 贝叶斯方法 贝叶斯推断的资料 书籍不少 比如 数理统计学简史 以及 统计决策论及贝叶斯分析 James O Berger著 等等 然介绍贝叶斯网络的中文资料则非常少 中文书籍总共
  • java设计学生类

    设计一个学生类 1 Student类中包含姓名 成绩两个属性 2分别给这两个属性定义两个方法 一个方法用于设置值 另一个方法用于获取值 3 Student类中定义一个无参的构造方法和一个接收两个参数的构造方法 两个参数分别为姓名和成绩属性赋
  • 制作Python,Raspberry Pi,电机和传感器版无线控制漫游车

    构建功能强大且可升级的个人机器人 您将学习如何控制伺服系统 响应传感器输入以及了解您的机器人在何处使用 GPS 您还将学习多种连接机器人并向其发送指令的方法 从 SSH 连接到从手机发送短信 内容 Raspberry Pi 介绍 Linux
  • 数字信息记忆-八大行星

    1 八大行星 水星 金星 地球 火星 木星 土星 天王星 海王星 2 提取关键字 水金球火木土天海 3 谐音联结 水晶 金 球喷火 烧木成土 从天落到海里
  • SQL语句基础练习(表的创建,基础查询)-提供建表语句和查询题目与解答

    说明 此基础练习的题目难度都比较简单 但是还有一些细节需要把握 此题目的主要目的是熟悉SQL语句的基础查询 关于建表方面的语句了解即可 因为一般开发中所用的都是图形化建表 SQL语句建表所用较少 虽然这些题目比较简单 但是开始中大多数都是这
  • Montery 无法接收12.1更新问题的处理

    Monterey 无法接收12 1更新问题的处理 说干货 打开App Store 搜索Monterey 点击macOS Monterey上的 查看 或者 获取 如果进度条卡住 重启系统再试
  • java zip压缩文件中文文件名乱码

    使用java util zipoutputstream发现中文名出现乱码 一直试着使用new String filename getBytes GBK ISO 8859 1 还是无效 后来查阅资料 java util zipoutputst
  • 字符串分割的几种方法

    方法一 利用STL自己实现split 函数 常用 简单 直观 原型 vector
  • 转 欢迎使用CSDN-markdown编辑器

    https blog csdn net CSDNwei article details 51836182 Extern的问题在于不知道这个关键词出现的时候到底是声明还是定义 谨记 声明可以多次 定义只能一次 函数的声明extern关键词是可