makefile
mkefile文件中定义了一系列的规则来指定,哪些文件需要线编译,哪些后编译,哪些需要重新编译,甚至进行更复杂的功能操作,因为makefile就像一个Shell脚本一样,其中也可以执行操作系统的命令。
mkefile带来的好处就是“自动化编译”,一旦写好,只需要一个make命令,整个工程完全自动编译,极大提高了软件开发的效率。
make是一个命令工具,是一个解释makefile中指令的命令工具,一般来说,大多数的IDE都有这个命令,比如visual c++的nmake,Linux下GNU的make,可见,makefile都成为了一种在工程方面的编译方法。
makefile文件中会使用gcc编译器对源代码进行编译,最终生成可执行文件或者库文件。
makefile文件的命令:makefile或者Makefile
makefile的工作原理
若想生成目标,检查规则中的所有的依赖条件是否都存在:
如果有的依赖条件不存在
向下搜索规则,看是否有生成该依赖条件的规则:
如果有规则用来生成该依赖条件,则执行规则中的命令生成依赖条件
如果没有规则用来生成该依赖文件,则报错
如果所有依赖条件都存在
检查规则中的目标是否需要更新,必须先检查它的所有依赖,依赖中有任何一个被更新,则目标必须更新(检查的规则是哪个时间段哪个更新)
- 如果目标的时间>依赖时间,不更新
- 如果目标的时间<依赖时间,则更新
makefile的基本规则
makefile由一组规则组成,规则如下:
目标: 依赖
(tab)命令
基本规则三要素
- 目标:要生成的目标文件
- 依赖: 目标文件由哪些文件生成
- 命令: 通过执行该命令由依赖文件生成目标
mkefile不同版本写法
现有main.c,fun1.c,fun2.c ,head.h三个文件
第一个版本
vi makefile
内容:
main:main.c fun1.c fun2.c
gcc -o main main.c fun1.c fun2.c
第二个版本
main:main.o fun1.o fun2.o
gcc -o main main.o fun1.o fun2.o
main.o:main.c
gcc -c main.c -I./
fun1.o:fun1.c
gcc -c fun1.c
fun2.o:fun2.c
gcc -c fun2.c
分步写,如果中间量更改,可以减少重新编译的内容
第三个版本
makefile的变量
在makefile中使用变量有点类似于C语言中的宏定义,使用该变量相当于内容替换,使用变量可以使makefile易于维护,修改起来简单
makefile有三种类型的变量
- 普通变量
- 自带变量
- 自动变量
普通变量
- 变量定义直接用 =
- 使用变量值用 $(变量名)
自带变量
makefile中也提供了一些变量(变量名大写)供用户直接使用,我们可以直接对其进行赋值
CC = gcc #arm-linux-gcc
CPPFLAGS:C预处理的选项有 -I
CFLAGS: C编译器的选项 -Wall -g -c
LDFLAGS: 链接器选项 -L -l
自动变量
- $@:表示规则中的目标
- $<:表示规则中的第一个条件
- $^:表示规则中所有条件,组成一个列表,以空格隔开,如果这个列表中有重复的项则消除重复项
自动变量只能在规则中的命令中使用
模式规则
至少在规则的目标定义中要包含%,%表示一个或多个,在依赖条件中同样可以使用%,依赖条件中的%取决于其目标
根据makefile的变量进行替换
利用变量进行更改,可以减少代码量
第四个版本
makefile函数
wildcard 查找指定目录下的指定类型的文件
src=$(wildcard *.c) //找到当前目录下所有后缀为.c的文件,赋值给src
patsubst 匹配替换
obj=$(patsubst %.c,%.o,$(src)) //把src变量里面所有后缀为.c的文件替换成.o
在makefile中所有的函数都是有返回值的
第五个版本
makefile的清理操作
用途:清楚编译生成的中间.o文件和最终目标文件
make clean 如果当前目录下有同名clean文件,则不执行clean对应的命令,解决方案:
伪目标声明:
.PHONY:clean
声明目标伪伪目标之后,makefile将不会检查该目标是否存在或者该目标是否需要更新
clean命令中的特殊符号
- “-”此条命令出错,make也会继续执行后续的命令
- “@”不显示命令本身,只显示结果。
其它
- make 默认执行第一个出现的目标,可通过make dest 指定要执行的目标
- make -f:-f执行一个makefile文件名称,使用make执行的指定的makefile:如make -f xxx(xxx是一个makefile文件,可以随意取名)
gdb调试
gdb介绍
GDB(GNU Debugger)是GCC的调试工具。其功能强大
GDB主要功能:
- 启动程序,可以按照你的自定义的要求随心所欲的运行程序
- 可让被调试的程序在你的自定义的要求随心所欲的运行程序
- 当程序被停止时,可以检查此时你的程序中所发生的事
- 动态的改变你程序的执行环境
生成调试信息
一般来说GDB主要调试的是C/C++的程序。要调试C/C++的程序,首先在编译时,我们需要把调试信息加入到可执行文件。使用编译器(cc/gcc/g++)的-g参数可以做到这一点。
如
gcc -g hello.c -o hello
如果没有-g,你将看不见程序的函数名、变量名,所代替的全是运行时的内存地址。
启动gdb
启动gdb : gdb program
program也就是你的执行文件,一般在当前目录下
设置运行参数
set args 可指定运行时参数。(如:set args 10 20 30 40 50)
显示源代码
GDB可以打印出所有调试程序的源代码,当然,在程序编译时一定要加上-g参数。
当程序停下来以后,可以通过list命令来打印程序的源代码
用法如下:
- list linenum :打印第linenum行的上下文内容
- list function:显示函数名为function的函数的源程序
- list:显示当前行后面的源程序
- list -:显示当前文件开始处的源程序
- list file:linenum:显示file文件下第linenum行
- list file:function:显示file文件的函数名为function的函数的源程序
一般打印当前行的上5行和下5行,默认是10行
使用下面的命令可以设置一次显示源程序的函数 - set listsize count :设置一次显示源代码的函数
- show listsize:查看当前listsize的设置
设置断点
简单断点-当前文件
break 设置断点,可以简写为b
b 10
意思是在源程序的第10行设置断点b func
在func函数入口处设置断点
多文件设置断点–其他文件
b filename:linenum
在源文件为filename的linenum行设置断点b filename:function
在源文件filename的function函数的入口处设置断点
查询所有断点
info b
== info break == i b
条件断点
为断点设置一个条件,使用关键字if
b xxx.c:n if intValue ==5
如果intValue的值等于5,在xxx.c文件第n行中设置断点
维护断点
delete [range…]
删除指定的断点
- delete 删除所有的断点
- delete num 删除断点为num的断点
- delete num1 num3 删除不连续的断点 num1 num3
- delete n-m 删除连续的断点,删除 n-m的断点
disable[range…]也可以简写为dis [range…]
指定断点无效,不会删除断点
enable[range…]也可以简写为ena[range…]
指定断点有效,对应解开disable设置的无效断点
调试代码
- run 运行程序,可简写为r
- next 单步跟踪,函数调用当做一条简单语句执行,可简写为n
- step 单步跟踪,函数调用进入被调用函数体内,可简写为s
- finish 退出进入的函数,如果出不去,可能是函数体中的循环有断点,可以将其删除或设置为无效
- untile 在一个循环体内单步跟踪,这个命令可以运行程序,直到退出循环体,可简写为u
- continue 继续运行程序,可简写为c(如果有断点,则调到下一个断点处)
查看变量的值
查看运行时变量的值
print 打印变量、字符串、表达式等的值,可简写为p
p count -----打印count的值
p &count ----打印count的地址
自动显示变量的值
可以设置一些自动显示的变量,当程序停住是,或在单步追踪时,这些变量会自动显示
相关的GDB命令是display
- display 变量名 设置自动显示
- info display --查看display设置的自动显示的信息
- undisplay num (info display时显示的编号)删除自动显示
- delete display dnums —删除自动显示,dums意味着自动显示的编号,如果要同时删除,用
空格
分隔,如果删除连续的,可以用-
- disable display dnums 设置dnums无效
- enable display dnums 设置dnums有效,与display相反
查看修改变量的值
ptype xxx —查看变量名为xxx的类型
p xxx ----打印变量名为xxx的值