C语言中头文件和源文件的关系

答案

#include预处理实际上就是完成了一个“复制代码并插入”的工作!

编译器工作阶段

1.预处理
2.词法和语法分析
3.编译(高级语言——汇编语言——二进制码——目标文件.obj)
4.连接(生成可执行文件.exe)

编译器各阶段工作工作过程

1.预处理阶段:
编译器以C文件作为一个单元,开始读C文件,读到包含头文件的语句时,从所有搜索路径寻找头文件。
找到后,处理宏,变量,函数声明,检测依赖关系,进行宏替换,检查是否有重定义和声明,最后将头文件中的东西扫描进当前C文件中,形成“中间C文件”。
2.编译阶段
将各个函数编译成二进制码,按照特定格式生成目标文件.obj
3连接阶段
将各个目标文件,连接最终生成可执行文件

为什么在头文件进行函数,变量和宏的声明,而在C文件进行变量定义,函数实现?

1.在头文件实现函数体的话,如果在多个C文件中引用,且同时编译多个C文件,那么在连接成可执行文件时,如果这段函数没有定义成局部函数,就会出现函数的多重定义错误

2.如果在头文件定义全局变量,并且赋初值。那么多个引用此头文件的C文件中存在相同变量名的copy,由于已经赋初值,所以每个copy都会分配一个空间,那么在最终连接阶段,会在DATA段存在多个相同变量;假设没有赋初值,则编译器会将这个变量放在BSS段,连接器对BSS段的多个同名变量仅分配一个存储空间。

3.如果在C文件中声明宏,结构体,函数等,那么在另一个C文件中引用相同的宏,结构体,就必须再做重复工作。且改动时需要分别都改动,不如放在头文件中统一改动。

4.在头文件中声明结构,函数等,当想把代码封装成库,又不想开源时,可以在头文件中写函数声明和用法。

5.使用头文件还可以避免函数的声明和定义的次序问题

静态链接和动态链接

静态链接 :.h+.lib(编译器将.cpp文件编译成.lib库文件,接口对外,实现代码不可见)
动态链接:.h+.cpp

你可能感兴趣的:(一些计算机常识)