Linux编译器-gcc/g++使用
- 编译器:gcc编译C语言,g++编译C语言/C++;
- gcc完成格式: gcc [选项] [要编译的文件] [选项] [目标文件]
- 程序的编译过程:预处理(进行宏替换),编译(生成汇编),汇编(生成机器可识别代码),链接(生成可执行文件或库文件)。
预处理:
1. 预处理功能主要包括宏定义,文件包含,文件编译,去注释等。
2. 预处理指令是以 # 号开头的代码行。
3. 实例:gcc -E hello.c -o hello.i
4. 选项 " -E " ,该选项的作用是让gcc在预处理结束后停止编译过程。
5. 选项 " -o " 是指目标文件," i "文件为已经预处理的C源程序
编译:
1. 在这个阶段中,gcc首先要检查代码的规范性、是否有语法错误等,以确定代码实际要做的工作,
在检查无误后,gcc把代码翻译成汇编语言。
2. 用户可以使用 “ -S-”选项进行查看,该选项进行进行编译而不进行汇编,生成汇编代码。
3. 实例:gcc -s hello.i -o hello.s
汇编:
1. 汇编阶段是把编译阶段生成的 “ .s ”文件转成目标文件
2. 读者在此可使用选项 " -c " 就可以看到汇编代码已转化为 " .o "的二进制目标代码了,
“ .o ”文件对标到win当中,就是目标程序(xxx.obj)
4. 实例:gcc -c hello.s -o hello.o
链接:
1.在成功编译之后,就进入了链接阶段,将多个目标和库链接到一起,生成可执行程序
2. 实例:gcc hello.o -o hello
-
在这里涉及到一个重要概念:库函数
我们在C程序中,并没有定义“ printf ”的函数实现,且在预编译中包含的“ stdio.h ”中也只有该函数的声明,而没有定义函数的实现,那么,是在哪里实现 “ printf ” 函数的呢?
事实上这些函数的实现都被系统放到名为 libc.so.6 的库文件中去了,在没有特别指定时,gcc会到系统默认的搜索路径“ /usr/lib ”下进行查找,也就是链接到 libc.so.6 库函数中去,这样就能实现函数 “ printf ”了,而这也就是链接的作用 -
函数库一般分为静态库和动态库两种
1. 静态库是指编译链接时,把库文件的代码全部加入可执行文件中,因此生成的文件比较大,但在运行时就不需要库文件了。其后缀名一般为“ .a ”
2. 动态库与之相反,在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为 “ .so”,如果前面所述的 libc.so.6 就是动态库。gcc在编译的时候默认使用动态库。完成了链接之后,gcc就可以生成可执行文件,如下所示: gcc hello.o -o hello
3. gcc默认生成的二进制程序,是动态链接,这点可以通过 file 命令验证。 -
gcc 选项
1. -E 中激活预处理,这个不生成文件,你需要把它重定向到一个输出文件里面 2. -S 编译到汇编语言不进行汇编和链接 3. -c编译到目标代码 4. -o文件输出到文件 5. -static此选项对生成的文件采用静态链接 6. -g生成调试信息。GNU调试器可利用该信息 7. -shared此选项将尽量使用动态库,所以生成文件比较小,但是需要系统由动态库。 8. -o0 9. -o1 10. -o2 11. -o3 编译器的优化选项的4个级别,-o0表示没有优化,-o1为缺省值,-o3优化级别最高 12. -w 不生成任何警告信息 13. -Wall生成所有警告信息