[C]#include和链接
概述
对于刚接触C语言的同学来说,通常对“在文件中用#include预处理操作符引入文件”和“编译时链接多个文件”这两个操作会有所混淆,这个文章主要为了解析一下它们的区别。
#include预处理操作符
对于此类操作,你可以在C语言预处理机制上去理解,预处理,顾名思义是编辑器在编译前进行的一系列操作,例如把预定义常量替换成字面量等等,具体可以去参考一下编译器对#define操作符的处理过程。
对于#include,实际上可以理解成PHP的include(如果你是先学PHP,再学C的同学),它实际上只是把文件引用进来,被引用文件被视为引用文件的一部分,文件作用域适用于这一个文件。
1.在文件a.h中声明了一个变量,a.h被b.c引用进来,这时a.h相当于b.c的一部分,所以b.c具备了a.h声明的变量
a.h:
int foo = ;
b.h
#include <stdio.h>
#include "a.h"int main(void)
{
printf("%d\n", foo);//输出1024
}
用编译器直接编译b.h,无需链接a.h,因为a.h已经被引入了:
gcc -std=gnu99 b.c -o b
2.为了证明被引入的文件会被视为跟引入文件一体这个事实,再用下面这个示例说明一下,b.c引入头文件a.h,a.h包含一个static变量,c.c包含试图访问a.h的变量的代码,但是c.c不引入a.h,而是编译时跟b.c链接:
我们先来看看正确的示范,首先不把foo声明为static编译程序,此时实际上是b.c和c.c相链(b.c包含了a.h)
a.h:
int foo = ;
b.h:
#include <stdio.h>
#include "a.h" extern void test(void);
int main(void)
{
printf("b:foo:%d\n", foo);
//test in c.c
test();
}
c.c
#include <stdio.h> extern int foo;
void test(void)
{
printf("c:foo:%d\n", foo);
}
编译程序:
gcc -std=gnu99 b.c c.c -o b
运行输出:
b:foo:
c:foo:
再来看看把头文件变量声明为static后的影响,此时把foo声明为static:
static int foo = ;
再次编译可见发现报错:
# gcc -std=gnu99 -trigraphs b.c c.c -o b
/tmp/cc8ojf4Q.o: In function `test':
c.c:(.text+0x6): undefined reference to `foo'
collect2: error: ld returned exit status
这是因为static具有文件作用域,b.c和c.c不视为同一个文件,而a.h和b.c则视为同一个文件;
可以看看以下示例,更进一步说明了这一点,a.h声明的常量又或者变量,都可以在b.c访问到,通过链接形式的c.c访问foo和BAR则不行:
a.h:
#define BAR 256
static int foo = ;
b.h:
#include <stdio.h>
#include "a.h" int main(void)
{
printf("b:bar:%d\n", BAR);//输出256
printf("b:foo:%d\n", foo);//输出1024
}
[C]#include和链接的更多相关文章
- linux下C/C++编译时系统搜索 include 和 链接库 文件路径的指定
C/C++程序在linux下被编译和连接时,GCC/G++会查找系统默认的include和link的路径,以及自己在编译命令中指定的路径.自己指定的路径就不说了,这里说明一下系统自动搜索的路径. ...
- Ubuntu Server(Ubuntu 14.04 LTS 64位)安装libgdiplus2.10.9出错问题记录
首先下载libgdiplus2.10.9安装包 wget http://download.mono-project.com/sources/libgdiplus/libgdiplus-2.10.9.t ...
- 《Entity Framework 6 Recipes》中文翻译系列 (23) -----第五章 加载实体和导航属性之预先加载与Find()方法
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-2 预先加载关联实体 问题 你想在一次数据交互中加载一个实体和与它相关联实体. ...
- c++ 中__declspec 的用法
__declspec ( extended-decl-modifier-seq )扩展修饰符:1:align(#) 用__declspec(align(#))精确控制用户自定数据的对齐方式 ,# ...
- Qt *.pro编写一般规则
qmake 之 CONFIG 与 QT 乱谈 看qtcn论坛中经常有人忘记 QT+=network 等语句.随便写写吧,或许对他人有帮助. 写来写去,发现越写越乱,就这样吧,反正主要的内容很简单. d ...
- __declspec关键字详细用法
__declspec关键字详细用法 __declspec用于指定所给定类型的实例的与Microsoft相关的存储方式.其它的有关存储方式的修饰符如static与extern等是C和C++语言的ANSI ...
- Visual C++ 设置适合自己的解决方案目录结构
Visual C++ 使用解决方案来管理项目,项目之间还可能有依赖关系,设置适合自己的解决方案目录结构,便于代码的管理.程序的发布. 下面开始一个虚拟解决方案设计: 假设此解决方案有应 ...
- 实现web数据同步的四种方式
http://www.admin10000.com/document/6067.html 实现web数据同步的四种方式 1.nfs实现web数据共享 2.rsync +inotify实现web数据同步 ...
- Dll学习(二)__declspec用法详解
__declspec用于指定所给定类型的实例的与Microsoft相关的存储方式.其它的有关存储方式的修饰符如static与extern等是C和 C++语言的ANSI规范,而__declspec是一种 ...
随机推荐
- Java多态之动态绑定
目录 Java多态之动态绑定 引用变量的类型 编译时类型 运行时类型 方法绑定 静态绑定 动态绑定 方法表 Java多态之动态绑定 上篇回顾:多态是面向对象程序设计非常重要的特性,它让程序拥有 更好的 ...
- 【Nodejs】375- 如何加快 Node.js 应用的启动速度
我们平时在开发部署 Node.js 应用的过程中,对于应用进程启动的耗时很少有人会关注,大多数的应用 5 分钟左右就可以启动完成,这个过程中会涉及到和集团很多系统的交互,这个耗时看起来也没有什么问题. ...
- 【JS】370- 总结异步编程的六种方式
点击上方"前端自习课"关注,学习起来~ 作者:Aima https://segmentfault.com/a/1190000019188824 众所周知 JavaScript 是 ...
- 【设计模式】代理模式-Proxy
转载:https://www.cnblogs.com/yangchongxing/p/7654725.html 代理模式定义如下: Provide a surrogate or placeholder ...
- webpack学习2.3webpack核心概念
核心概念(四个) Entry(入口) Output(出口) Loaders()来处理其他类型的资源文件 Plugins(插件) 1.入口(Entry) 作用:代码的入口,打包的入口,单个或多个, 示例 ...
- 时间轮算法(TimingWheel)是如何实现的?
前言 我在2. SOFAJRaft源码分析-JRaft的定时任务调度器是怎么做的?这篇文章里已经讲解过时间轮算法在JRaft中是怎么应用的,但是我感觉我并没有讲解清楚这个东西,导致看了这篇文章依然和没 ...
- Table实现表头固定 内容滚动
<div style="width: 800px;"> <div class="table-head"> <table> & ...
- 《Dotnet9》建站-建站20天感悟
时间如流水,只能流去不流回! 点赞再看,养成习惯,这是您给我创作的动力! 本文 Dotnet9 https://dotnet9.com 已收录,站长乐于分享dotnet相关技术,比如Winform.W ...
- [ASP.NET Core 3框架揭秘] 依赖注入[10]:与第三方依赖注入框架的适配
.NET Core具有一个承载(Hosting)系统,承载需要在后台长时间运行的服务,一个ASP.NET Core应用仅仅是该系统承载的一种服务而已.承载系统总是采用依赖注入的方式来消费它在服务承载过 ...
- C# 中的栈和堆
程序运行时,它的数据必须存储在内存中.一个数据项需要多大的内存.存储在内存中的什么位置.以及如何存储都依赖于该数据项的类型. 运行中的程序使用两个内存区域来存储数据:栈和堆. 栈 栈是一个内存数组,是 ...