1.gcc/g++的使用
c程序的翻译过程分为如下几步:
-
预处理(-E):头文件的展开,去注释,宏替换,条件编译
预处理完成会生成".i " 文件。通过观察文件大小我们就可以看出来在预处理过程中做了一系列的操作
-
编译(-S):C代码编译成汇编语言
-
汇编(-c):将汇编语言翻译为目标文件(变为二进制)
-
链接:生成可执行文件
这个绿色的就是可执行程序。
在这里涉及到一个重要的概念:函数库
- 我们的C程序中,并没有定义“printf”的函数实现,且在预编译中包含的“stdio.h”中也只有该函数的声明,而没有定义函数的实现,那么,是在哪里实“printf”函数的呢?
- 最后的答案是:系统把这些函数实现都被做到名为 libc.so.6 的库文件中去了,在没有特别指定时,gcc 会到系统默认的搜索路径“/usr/lib”下进行查找,也就是链接到 libc.so.6 库函数中去,这样就能实现函数“printf”了,而这也就是链接的作用
2.动态库、静态库
函数库一般分为静态库和动态库两种
-
静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了。其后缀名一般为“.a”
-
动态库与之相反,在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为“.so”,
-
gcc默认生成的二进制程序,是动态链接的,这点可以通过 file 命令验证。
不过我们可以通过 gcc 带上 -static,变成静态链接
通过观察文件的大小我们也可以看出静态链接是将库文件的代码拷贝到可执行文件中
3.Linux调试器-gdb的使用、
我们直到程序的发布方式有如下两种
- debug --程序本身会被加入更多的调试信息,以便于进行调试
- release --不会添加任何调试信息,不可被调试
在centos7 gcc中默认生成的可执行程序是release版本的,不可以被调试
如果想发布debug版本,必须在源代码生成二进制程序的时候, 加上 -g 选项
通过观察,我们也可以发现debug版本比release版本更大一些
下面我们介绍一些gdb的操作指令
- list/l 行号:显示binFile源代码,接着上次的位置往下列,每次列10行。
- list/l 函数名:列出某个函数的源代码。
- r或run:运行程序。
- n 或 next:单条执行。(逐过程)
- s或step:进入函数调用(逐语句)
- break(b) 行号:在某一行设置断点
- break 函数名:在某个函数开头设置断点
- info break :查看断点信息。
- finish:执行到当前函数返回,然后挺下来等待命令
- print§:打印表达式的值,通过表达式可以修改变量的值或者调用函数
- continue(或c):从当前位置开始连续而非单步执行程序
- display 变量名:跟踪查看一个变量,每次停下来都显示它的值
- undisplay:取消对先前设置的那些变量的跟踪
- until X行号:跳至X行
- set var:修改变量的值
- quit:退出gdb
- breaktrace(或bt):查看各级函数调用及参数(看栈帧)
:取消对先前设置的那些变量的跟踪 - until X行号:跳至X行
- set var:修改变量的值
- quit:退出gdb
- breaktrace(或bt):查看各级函数调用及参数(看栈帧)
- info(i) locals:查看当前栈帧局部变量的值