本篇文章进行gcc/g++的学习!!!
预处理:宏替换、展开头文件、去注释、条件编译
编译:检查语法,将源文件翻译成汇编语言
汇编:将汇编语言翻译成"可重定位二进制文件"
链接:链接动静态库、头文件,生成可执行文件/库文件
- 格式:gcc [选项] 待编译的源文件 [选项] [目标文件]
预处理功能主要包括宏定义、文件包含、条件编译、去注释等等…
预处理指令是以#号开头的代码行
实例:gcc –E hello.c –o hello.i
选项“-E”,该选项的作用是让 gcc 在预处理结束后停止编译过程。
选项“-o”是指目标文件,“.i”文件为已经过预处理的C原始程序
[lyh_sky@localhost lesson9]$ cat hello.c
#include
#define N 100
int main()
{
//printf("hello world\n");
//printf("hello world\n");
//printf("hello world\n");
printf("hello world\n");
#ifdef DEBUG
printf("hello debug\n");
#else
printf("hello release\n");
#endif
return 0;
}
// 从现在开始进行程序的翻译,当进行到预处理结束后就停下来
[lyh_sky@localhost lesson9]$ gcc -E hello.c -o hello.i
[lyh_sky@localhost lesson9]$ ls
hello.c hello.i
观察下面预处理后和源代码,发现宏已经替换,注释已经去掉,条件编译后
在这个阶段中,gcc 首先要检查代码的规范性、是否有语法错误等,以确定代码的实际要做的工作,在检查无误后,gcc 把代码翻译成汇编语言
用户可以使用“-S”选项来进行查看,该选项只进行编译而不进行汇编,生成汇编代码
实例:gcc –S hello.i –o hello.s
[lyh_sky@localhost lesson9]$ ls
hello.c hello.i
// 从现在开始进行程序的翻译,当进行到编译结束后就停下来
[lyh_sky@localhost lesson9]$ gcc -S hello.i -o hello.s
[lyh_sky@localhost lesson9]$ ls
hello.c hello.i
[lyh_sky@localhost lesson9]$ gcc -S hello.i -o hello.s
观察以下文件,预处理文件已经变成汇编代码
// 从现在开始进行程序的翻译,当进行到汇编结束后就停下来
[lyh_sky@localhost lesson9]$ gcc -c hello.s -o hello.o
[lyh_sky@localhost lesson9]$ ls
hello.c hello.i hello.o hello.s
观察以下文件,汇编代码已经完全变成二进制机器码
在成功编译之后,就进入了链接阶段
实例:gcc hello.o –o hello
// 直接生成可执行程序后停止
[lyh_sky@localhost lesson9]$ gcc hello.o -o hello
[lyh_sky@localhost lesson9]$ ls
hello hello.c hello.i hello.o hello.s
[lyh_sky@localhost lesson9]$ ls
hello hello.c hello.i hello.o hello.s
[lyh_sky@localhost lesson9]$ ./hello.o
bash: ./hello.o: 权限不够
[lyh_sky@localhost lesson9]$ chmod u+x hello.o
[lyh_sky@localhost lesson9]$ ls -l hello.o
-rwxrw-r--. 1 lyh_sky lyh_sky 1568 10月 30 00:39 hello.o
[lyh_sky@localhost lesson9]$ ./hello.o
bash: ./hello.o: 无法执行二进制文件
为什么无法执行目标文件呢?
我们的C程序中,并没有定义“printf”的函数实现,且在预编译中包含的“stdio.h”中也只有该函数的声明,而没有定义函数的实现,那么,是在哪里实“printf”函数的呢?
答案是:系统把这些函数实现都被做到名为 libc.so.6 的库文件中去了,在没有特别指定时,gcc 会到系统默认的搜索路径“/usr/lib”下进行查找,也就是链接到 libc.so.6 库函数中去,这样能实现函数“printf”了,而这也就是链接的作用
ls /lib64/libc*
静态库是指编译链接时,把库文件的代码全部拷贝到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了,在Linux后缀名一般为[.a]
动态库与之相反,在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为[.so],如前面所述的 libc.so.6 就是动态库
gcc默认生成的二进制程序,是动态链接的,这点可以通过 file 命令验证
动态库的优缺点:
静态库的优缺点:
gcc选项记忆:esc(键盘左上角) iso(镜像文件)