概述

对于刚接触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和链接的更多相关文章

  1. linux下C/C++编译时系统搜索 include 和 链接库 文件路径的指定

     C/C++程序在linux下被编译和连接时,GCC/G++会查找系统默认的include和link的路径,以及自己在编译命令中指定的路径.自己指定的路径就不说了,这里说明一下系统自动搜索的路径.   ...

  2. 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 ...

  3. 《Entity Framework 6 Recipes》中文翻译系列 (23) -----第五章 加载实体和导航属性之预先加载与Find()方法

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-2  预先加载关联实体 问题 你想在一次数据交互中加载一个实体和与它相关联实体. ...

  4. c++ 中__declspec 的用法

    __declspec ( extended-decl-modifier-seq )扩展修饰符:1:align(#)    用__declspec(align(#))精确控制用户自定数据的对齐方式 ,# ...

  5. Qt *.pro编写一般规则

    qmake 之 CONFIG 与 QT 乱谈 看qtcn论坛中经常有人忘记 QT+=network 等语句.随便写写吧,或许对他人有帮助. 写来写去,发现越写越乱,就这样吧,反正主要的内容很简单. d ...

  6. __declspec关键字详细用法

    __declspec关键字详细用法 __declspec用于指定所给定类型的实例的与Microsoft相关的存储方式.其它的有关存储方式的修饰符如static与extern等是C和C++语言的ANSI ...

  7. Visual C++ 设置适合自己的解决方案目录结构

    Visual C++ 使用解决方案来管理项目,项目之间还可能有依赖关系,设置适合自己的解决方案目录结构,便于代码的管理.程序的发布. 下面开始一个虚拟解决方案设计:         假设此解决方案有应 ...

  8. 实现web数据同步的四种方式

    http://www.admin10000.com/document/6067.html 实现web数据同步的四种方式 1.nfs实现web数据共享 2.rsync +inotify实现web数据同步 ...

  9. Dll学习(二)__declspec用法详解

    __declspec用于指定所给定类型的实例的与Microsoft相关的存储方式.其它的有关存储方式的修饰符如static与extern等是C和 C++语言的ANSI规范,而__declspec是一种 ...

随机推荐

  1. Java多态之动态绑定

    目录 Java多态之动态绑定 引用变量的类型 编译时类型 运行时类型 方法绑定 静态绑定 动态绑定 方法表 Java多态之动态绑定 上篇回顾:多态是面向对象程序设计非常重要的特性,它让程序拥有 更好的 ...

  2. 【Nodejs】375- 如何加快 Node.js 应用的启动速度

    我们平时在开发部署 Node.js 应用的过程中,对于应用进程启动的耗时很少有人会关注,大多数的应用 5 分钟左右就可以启动完成,这个过程中会涉及到和集团很多系统的交互,这个耗时看起来也没有什么问题. ...

  3. 【JS】370- 总结异步编程的六种方式

    点击上方"前端自习课"关注,学习起来~ 作者:Aima https://segmentfault.com/a/1190000019188824 众所周知 JavaScript 是  ...

  4. 【设计模式】代理模式-Proxy

    转载:https://www.cnblogs.com/yangchongxing/p/7654725.html 代理模式定义如下: Provide a surrogate or placeholder ...

  5. webpack学习2.3webpack核心概念

    核心概念(四个) Entry(入口) Output(出口) Loaders()来处理其他类型的资源文件 Plugins(插件) 1.入口(Entry) 作用:代码的入口,打包的入口,单个或多个, 示例 ...

  6. 时间轮算法(TimingWheel)是如何实现的?

    前言 我在2. SOFAJRaft源码分析-JRaft的定时任务调度器是怎么做的?这篇文章里已经讲解过时间轮算法在JRaft中是怎么应用的,但是我感觉我并没有讲解清楚这个东西,导致看了这篇文章依然和没 ...

  7. Table实现表头固定 内容滚动

    <div style="width: 800px;"> <div class="table-head"> <table> & ...

  8. 《Dotnet9》建站-建站20天感悟

    时间如流水,只能流去不流回! 点赞再看,养成习惯,这是您给我创作的动力! 本文 Dotnet9 https://dotnet9.com 已收录,站长乐于分享dotnet相关技术,比如Winform.W ...

  9. [ASP.NET Core 3框架揭秘] 依赖注入[10]:与第三方依赖注入框架的适配

    .NET Core具有一个承载(Hosting)系统,承载需要在后台长时间运行的服务,一个ASP.NET Core应用仅仅是该系统承载的一种服务而已.承载系统总是采用依赖注入的方式来消费它在服务承载过 ...

  10. C# 中的栈和堆

    程序运行时,它的数据必须存储在内存中.一个数据项需要多大的内存.存储在内存中的什么位置.以及如何存储都依赖于该数据项的类型. 运行中的程序使用两个内存区域来存储数据:栈和堆. 栈 栈是一个内存数组,是 ...