概述

对于刚接触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. 最全的三剑客和linux正则符号讲解

    第2章 linux符号系列与三剑客    194 2.1 特殊符号系列    194 2.1.1 普通符号系列    194 2.1.2 引号符号系列    196 2.1.3 定向符号    197 ...

  2. Ceph 块存储

    任何普通的linux主机都可以充当ceph客户机,客户机通过网络与ceph存储集群交互以存储或检索用户数据.Ceph RBD支持已经添加到linux主线内核中,从2.6.34以及以后版本开始. === ...

  3. 玩转摄像头之 基于SDRAM缓冲 USB2.0视频采集系统 MT9T001、MT9P031 演示 展示

    玩转摄像头之  基于SDRAM缓冲 USB视频采集系统  MT9T001.MT9P031 最新设计的系统: 核心板(FPGA+SDRAM)+底板(68013+DVP)+sensor 先看图 核心板 正 ...

  4. HTML中用AJAX方式把数据存储到浏览器中并取出

    把对象只有转成字符串形式才可以存入,取出则是把字符串转成对象

  5. 【面试】386- JavaScript 面试 20 个核心考点

    点击上方"前端自习课"关注,学习起来~ 引言 Javascript是前端面试的重点,本文重点梳理下 Javascript 中的常考基础知识点,然后就一些容易出现的题目进行解析.限于 ...

  6. Docker swarm实战总结

    一.简介 Swarm 是 Docker 官方提供的一款集群管理工具,其主要作用是把若干台 Docker 主机抽象为一个整体,并且通过一个入口统一管理这些 Docker 主机上的各种 Docker 资源 ...

  7. Appium移动端自动化测试--控件定位方法

    常用定位手段 id Accessibility ID XPath 控件基础知识 DOM: Document Object Model文档对象模型 DOM应用:最早应用于HTML和Javascript的 ...

  8. django基础之day04,聚合查询和分组查询

    聚合查询: 聚合函数必须用在分组之后,没有分组其实默认整体就是一组 Max Min Sum Avg Count 1.分组的关键字是:aggretate 2.导入模块 from django.db.mo ...

  9. jdk13-新特性预览

    一新特性 350: Dynamic CDS Archives(动态CDS档案) 351: ZGC: Uncommit Unused Memory(ZGC:取消提交未使用的内存) 353: Reimpl ...

  10. create-react-app 超级慢的解决方法

    create-react-app超级慢的解决方法 在操作官方实例Create React App时,需要执行指令: create-react-app my-app 来创建一个新的React应用.由于某 ...