Linux第6章 常用开发工具.ppt
第第6 6章章 常用开发工具常用开发工具6.1 概述概述vLinux开发工作经常是在Linux用户决定共同完成一个项目时开始的。当开发工作完成后,该软件就被放到Internet站点上,任何用户都可以访问和下载它。v大多数Linux软件是经过自由软件基金会(Free Software Foundation)提供的GNU(GNU 即 GNUs not UNIX)公开认证授权的,因而通常被称作GNU软件。vGNU软件免费提供给用户使用,并被证明是非常可靠和高效的。许多流行的Linux实用程序,如:C编译器、shell和编辑器都是GNU软件应用程序。6.2 gcc编译系统编译系统v目前,linux平台上最常用的是C语言,其编译系统是gcc,能够编译用C,C+等语言编写的程序。v一般来说,系统安装后就已经安装和设定好了gcc。v在shell的提示符下键入gcc v,屏幕上就会显示出目前正在使用的gcc的版本。6.2.1 C语言编译过程 vC语言程序包括:源文件、头文件、库文件;在Linux系统中,C/C+程序编译命令是gcc;v当使用gcc时,gcc会完成预处理、编译、汇编和连接预处理、编译、汇编和连接;前三步生成目标文件,连接时把生成的目标文件链接成可执行文件;vgcc可以针对不同的源程序文件进行不同处理,文件格式以文件的后缀来识别,常见的如表6.1所示。1 1预处理阶段预处理阶段v预处理是常规编译之前预先进行的工作,故此得名。v负责读取C语言源文件,对其中以“#”开头的指令(伪指令)和特殊符号进行处理,如:a.将将“#include”所指出的文件替代该程序行,有两种格式:所指出的文件替代该程序行,有两种格式:b.#include c.预处理程序在预处理程序在/usr/include目录下找文件目录下找文件d.#include“文件名文件名”e.首先在当前工作目录中找,然后到标准目录首先在当前工作目录中找,然后到标准目录/usr/include中找;中找;f.备注:备注:g.使用使用gcc命令时设置选项,指定查找头文件时要优先搜索的目录。命令时设置选项,指定查找头文件时要优先搜索的目录。b.对对C语言源程序中的宏名进行宏替换。语言源程序中的宏名进行宏替换。c.例:例:#define EOF-1d.预处理程序将程序中有预处理程序将程序中有EOF的部分以的部分以-1取代。取代。宏定义:1.可以在C程序中:#define name value:如:#define stuname“Wang”2.也可以在gcc命令的选项中设置宏定义;如:gcc D name=definition第二种方式的优先级高于第一种方式,可以覆盖源文件中的定义。v预处理程序对源程序进行“替换”之后,输出的文件就不包含宏定义、文件包含、条件编译等指令,与源文件功能相同,而形式不同。gcc命令的使用v在Linux系统中,C/C+程序编译命令是gcc,例如:$gcc options filenames1.其中filenames为所要编译的程序源文件;2.执行完成后,生成默认的可执行文件a.out;3.options部分可以有较多取值,如:预处理选项、编译选项、优化选项、连接选项,使得gcc命令的功能很多。【例例】gcc预处理选项预处理选项$cathello.c#include“test1.h”#definevar1“callforhelp”main()printf(“displayDvariable%sn”,DOPTION);printf(“displayoverwritevar1=%sn”,var1);printf(“hello,everyone!n”);假设上述程序中,头文件test1.h存放在目录/temp中,且头文件里定义了变量var1,下面用gcc命令对上述C程序进行编译,$gcchello.c则会提示找不到头文件test1.h,以及DOPTION未定义;宏定义宏定义$gccI/temphello.c因此,编译的时候要在gcc命令的选项里面,加入头文件test1.h的路径:此时,会提示:在gcc命令的选项里加入对DOPTION的宏定义:变量var1重定义、DOPTION未定义$gccI/tempDDOPTION=”test”Ehello.c只做预处理,比如:宏替换,用参数的取值替代宏名;只做预处理,比如:宏替换,用参数的取值替代宏名;不做编译,将结果显示在标准输出上。不做编译,将结果显示在标准输出上。main()printf(“displayDvariable%sn”,“test”);printf(“displayoverwritevar1=%sn”,“callforhelp”);printf(“hello,everyone!n”);此时,若要用gcc命令编译并执行hello.c程序,则去掉-E选项,编译完成后,生成默认的可执行文件a.out$a.outdisplayDvariabletestdisplayoverwritevar1=callforhelphello,everyone!2 2编译阶段编译阶段v对预处理之后的输出文件进行词法分析、语法分析,试图找出所有不符合语法规则的部分。v并根据问题给出错误消息,终止编译,或给出警告。v当确定程序符合语法规则后,将其“翻译”为功能等价的中间代码,或汇编代码。3 3汇编过程汇编过程v汇编程序(Assembler)把汇编代码翻译成目标机器代码;v包括代码段和数据段等部分,前者包括程序指令,后者存放各种全局或局部变量。2gcc的编译程序选项常用选项及其作用 选选 项项 格格 式式 功功 能能-c 只生成目标文件,不进行连接。用于对源文件的分别编译只生成目标文件,不进行连接。用于对源文件的分别编译-S 只进行编译,不做汇编,生成汇编代码文件格式,其名与源只进行编译,不做汇编,生成汇编代码文件格式,其名与源文件相同,但扩展名为文件相同,但扩展名为.s-o file 将输出放在文件将输出放在文件file中。如果未使用该选项,则可执行文件放中。如果未使用该选项,则可执行文件放在在a.out中中-g 指示编译程序在目标代码中加入供调试程序指示编译程序在目标代码中加入供调试程序gdb使用的附加使用的附加信息信息-v 在标准出错输出上显示编译阶段所执行的命令,即编译驱动在标准出错输出上显示编译阶段所执行的命令,即编译驱动程序及预处理程序的版本号程序及预处理程序的版本号$catm1.c#includemain()intr;printf(“enteranintegern”);scanf(“%d”,&r);square(r);return0;【例例】gcc编译选项编译选项$catm2.c#includeintsquare(intx)printf(“square=%dn”,x*x);return(x*x);若直接编译m1.c文件:gcc m1.c,则会提示:m1.c文件中的main函数调用的square函数,但没有事先定义和声明。因此需要使用-c选项$gcccm1.c$gcccm2.c$gccm1.om2.oom12$m12enteraninteger6square=36-c-c选项表示:选项表示:只生产目标文件(后缀为只生产目标文件(后缀为.o.o,参,参见表见表6.16.1),而不进行连接,可),而不进行连接,可用于对源文件分别编译。用于对源文件分别编译。4 4连接阶段连接阶段v连接程序(Linker)要解决外部符号访问地址问题,即:将一个文件中引用的符号(如:变量、函数调用),与该符号在另外一个文件中的定义连接起来,最终成为操作系统可以执行的可执行文件。6.3 gdb程序调试工具v程序中的错误可按性质分为三种:(1)编译错误,即语法错误。在编译阶段出现,如:括号不对称、缺少分号等;(2)运行错误:运行时才能发现,如:除数为0,循环终止条件无法达到。(3)逻辑错误:程序可以正常运行,但结果不对。查找程序中的错误,诊断其准确位置,并予以改正,这就是程序调试。vLinux系统中包含了调试程序gdb,它是一个用来调试C和 C+程序的调试器;vgdb可以在程序运行时观察程序的内部结构和内存的使用情况;vgdb 所提供的一些功能如下所示:运行程序,设置程序运行的参数和环境;控制程序在指定的条件下停止运行;当程序停止时,可以检查程序的状态;动态监视程序中变量的值;6.3.1 启动gdb和查看内部命令gdb程序调试的对象是可执行文件,而不是程序的源代码文件;如果要让产生的可执行文件可以用来调试,需在执行gcc指令编译程序时,加上-g参数,指定程序在编译时包含调试信息;(P181,表6.3)并等待用户输入相应的内部命令并等待用户输入相应的内部命令 6.3.2 应用示例下面的这段程序有错误,以其为例,显示gdb调试程序的一般情况。宏定义,变量宏定义,变量BIGNUM值值为为1000调用函数调用函数index_m时,时,将将intary,fltary两个数组两个数组的首地址,作为参数传递的首地址,作为参数传递给函数。给函数。理论分析intary数组数组分配的分配的内存块内存块fltary数组数组分配的分配的内存块内存块内存块大小为内存块大小为100个个int型数据型数据所占用的空间所占用的空间内存块大小为内存块大小为100个个 float 型数据型数据所占用的空间所占用的空间如果进行1000次循环和赋值操作,可能发生什么情况?传递给传递给index_m函数的是数组首地址函数的是数组首地址ary数组数组分配的分配的内存块内存块fary数组数组分配的分配的内存块内存块可能发生的情况:如果fltary数组在内存中的空间位于intary数组前面,则:?当进行到第当进行到第101次次循环,也就是循环,也就是i=100的时候,的时候,fary100在内存在内存中的地址与中的地址与ary0的地址相同!的地址相同!如何通过如何通过gdb调试工具验证上述错误?调试工具验证上述错误?(1 1)在使用在使用在使用在使用gccgcc编译的时候,需要保留一些信息编译的时候,需要保留一些信息编译的时候,需要保留一些信息编译的时候,需要保留一些信息(如:变量的值,数组的内存地址等),(如:变量的值,数组的内存地址等),(如:变量的值,数组的内存地址等),(如:变量的值,数组的内存地址等),才能进行才能进行才能进行才能进行gdbgdb调试:方法如下:调试:方法如下:调试:方法如下:调试:方法如下:使用带-g选项的gcc命令对该程序进行编译:$gcc -g dbme.c -o dbme (2 2)用可执行程序文件名用可执行程序文件名用可执行程序文件名用可执行程序文件名dbmedbme作为参数,启动作为参数,启动作为参数,启动作为参数,启动gdbgdb。$gdb dbme 即可进入即可进入gdb环境如下图所示环境如下图所示初始化完成后,回到初始化完成后,回到gdb提示符状态提示符状态(3)进入)进入gdb环境后,用环境后,用run命令运行该程序,系统给出错误提示命令运行该程序,系统给出错误提示 1.gdb在执行过程中,收到系统发送的在执行过程中,收到系统发送的SIGSEGV信号,则停止运行,表明源程序中出现了段错误,信号,则停止运行,表明源程序中出现了段错误,即:访问了错误的内存段。即:访问了错误的内存段。2.该段错误发生在源文件该段错误发生在源文件dbme.c的第的第19行中,行中,index_m函数中;函数中;并且,并且,gdb给出了两个数组给出了两个数组ary,fary的基地址,的基地址,faryfary数组的基地址小于数组的基地址小于数组的基地址小于数组的基地址小于aryary数组的基地址。数组的基地址。数组的基地址。数组的基地址。使用使用backtrace命令,显示函数调用的时,用户栈的情况。命令,显示函数调用的时,用户栈的情况。v参照源码中错误行的上下文,使用参照源码中错误行的上下文,使用list命令显示相关行的内容命令显示相关行的内容栈底是最初执行的函数,栈底是最初执行的函数,即即main()函数函数栈顶是当前正在执行的函数,说明执行栈顶是当前正在执行的函数,说明执行到该函数时,出现错误被停止。到该函数时,出现错误被停止。List命令不带参数时,显示当前行命令不带参数时,显示当前行的上下的上下5行,总共行,总共10行。行。用用break命令设置断点,设置当执行到某一行时停止运行;命令设置断点,设置当执行到某一行时停止运行;并且可以结合并且可以结合step命令,一行行跟踪程序执行过程命令,一行行跟踪程序执行过程设置断点设置断点程序的第程序的第19行,且行,且i=100时,程序停止运行时,程序停止运行step命令,每次执行一行命令,每次执行一行当当i=100时,时,ary0的值发生了错误的值发生了错误且此时,且此时,fary100的地址与的地址与ary的基地址的基地址一样,发生了冲突。一样,发生了冲突。当当i=100时,时,ary0的值发生了错误,的值发生了错误,fary100的值是对的的值是对的另外设置一个断点,当执行到第另外设置一个断点,当执行到第19行时,且行时,且i值等于值等于99,停止运行停止运行此时数组中的值是正确的。此时数组中的值是正确的。此时数组中的值是正确的。此时数组中的值是正确的。v可见,数组的大小与循环体中变量i的变化范围有矛盾!1001000(BIGNUM的值)vfary数组的100号元素占用的内存地址,与ary数组0号元素占用的内存地址相同。结论:v准备工作:准备工作:为了发挥为了发挥gdb的全部功能,需要在编译源程序时使用的全部功能,需要在编译源程序时使用-g选项选项:$gcc-gm1.c-om1v启动启动gdb的方法有以下几种:的方法有以下几种:(1)直接使用)直接使用shell命令命令gdb$gdb (2)以一个可执行程序作为)以一个可执行程序作为gdb的参数的参数$gdb m1 gdb调试过程中的常用命令调试过程中的常用命令归纳与总结归纳与总结归纳与总结归纳与总结一、显示源程序和数据1 1显示和搜索源程序显示和搜索源程序在被调试的源程序中,进行上下文搜索,也可设定搜索路径。(1)显示源文件 利用list命令可以显示源文件中指定的函数或代码行;P186,表6.6(2)模式搜索:在源代码中搜索给定模式的命令 表6.7 forward-search regexp search regexp reverse-search regexp 归纳与总结归纳与总结归纳与总结归纳与总结2 2查看运行时数据查看运行时数据查看运行时数据查看运行时数据(1)print命令命令 一般使用格式是一般使用格式是:print /fmt expv当被调试的程序停止时,可以用当被调试的程序停止时,可以用print命令,查看当前程序中运行的数据。命令,查看当前程序中运行的数据。如:如:print i print i*j(2)gdb所支持的运算符所支持的运算符 type adrexp 表示一个数据类型为表示一个数据类型为type、存放地址为、存放地址为adrexp的数据。的数据。运算符:运算符:print array10 从基地址从基地址array开始的开始的10个数组元素值个数组元素值 print array35 从从array第三个元素开始的,第三个元素开始的,5个数组元素值个数组元素值 file:var (或者(或者 function:var)表示文件表示文件file(或者函数(或者函数function)中变量)中变量var的值的值 归纳与总结归纳与总结归纳与总结归纳与总结二、控制程序的执行u进入gdb后,可以在源程序的某些行上设置断点(breakpoint)程序执行到断点所在行,则暂停执行,gdb显示函数调用的踪迹和变量值;用户可以根据需要,自行添加、删除断点。此外还有:u观察点(watchpoint):观察某个表达式的值是否发生变化u捕捉点(catchpoint):针对程序运行时出现的事件,如:进程的创建断点、观察点、捕捉点统称为停止点。归纳与总结归纳与总结归纳与总结归纳与总结1 1设置和显示断点设置和显示断点设置和显示断点设置和显示断点(1)设置断点:用)设置断点:用break命令设置断点:命令设置断点:P190vbreak linenum vbreak linenum if condition vbreak function vbreak file:linenumvbreak file:function vbreak *address vbreak (2)显示断点:显示程序中设置了哪些断点)显示断点:显示程序中设置了哪些断点vinfo breakpoints numvinfo break num归纳与总结归纳与总结归纳与总结归纳与总结 vv维护停止点:清除和停用停止点维护停止点:清除和停用停止点维护停止点:清除和停用停止点维护停止点:清除和停用停止点 delete clear disable enable vv运行程序:设置断点后,用运行程序:设置断点后,用运行程序:设置断点后,用运行程序:设置断点后,用runrun命令运行程序命令运行程序命令运行程序命令运行程序归纳与总结归纳与总结归纳与总结归纳与总结vv程序的单步跟踪程序的单步跟踪程序的单步跟踪程序的单步跟踪 设置断点后,可以让程序一步步地向下执行,用户可以设置断点后,可以让程序一步步地向下执行,用户可以仔细检查运行过程,实行单步跟踪的命令是仔细检查运行过程,实行单步跟踪的命令是step和和next,其格式是:其格式是:step N 其中其中N为步长为步长 next N 两者区别是:两者区别是:两者区别是:两者区别是:后者遇到函数调用时,执行整个函数,即将其作为一条指后者遇到函数调用时,执行整个函数,即将其作为一条指后者遇到函数调用时,执行整个函数,即将其作为一条指后者遇到函数调用时,执行整个函数,即将其作为一条指令对待;令对待;令对待;令对待;前者进入函数内执行,每次仍然是执行前者进入函数内执行,每次仍然是执行前者进入函数内执行,每次仍然是执行前者进入函数内执行,每次仍然是执行N N行语句。行语句。行语句。行语句。归纳与总结归纳与总结归纳与总结归纳与总结三、其他常用命令1执行shell命令:在gdb环境中,执行Linux的shell命令 其格式是:shell command-string 如:(gdb)shell date 二 8月 22 20:22:48 CST 2006 (gdb)2修改变量值:用户根据需要更改程序运行路线、变量的值,如:(gdb)print x=10(gdb)set variable x=103跳转执行跳转执行通常,被调试程序是顺序执行的,可以利用通常,被调试程序是顺序执行的,可以利用jump命令,命令,在在gdb环境中让程序跳转到指定的代码行。格式为:环境中让程序跳转到指定的代码行。格式为:jump linenumjump *addr 代码行的内存地址代码行的内存地址6.3 程序维护工具makeu软件开发过程中,往往采用结构化的程序设计思想,将一个大型程序分为若干个功能明确的子程序;u最终的可执行文件依赖于各个目标文件、源文件、库文件等,如果其中某些文件修改了,那么是否需要把所有文件都重新编译、连接一遍呢?u为了减轻系统的编译负担,Linux开发环境提供了程序维护工具:make6.3.1 make的工作机制v通过使用make工具,程序员只需要定义各文件之间的依赖关系和相关操作,make工具自动完成产生新版本的必须操作。v其主要功能是:执行生成新版本目标程序的各个步骤,即:自动检测一个大型程序的哪些部分需要重新编译,然后发出编译命令。基本原理基本原理 1.要使用make命令,必须编写一个叫做makefile的文件,这个文件描述了软件包中文件之间的关系,提供更新每个文件的命令;2.一般在一个软件包里,通常是可执行文件靠目标文件(.o后缀)来更新,目标文件靠编译源文件来更新;3.makefile写好之后,每次改变了某些源文件,只要执行make命令,所有必要的重新编译将执行。4.make程序利用makefile中的数据、每个文件的最后修改时间来确定那个文件需要更新,对于需要更新的文件,根据makefile数据中定义的命令来更新。是一个文本形式的数据库文件,其中包含一些规则,告诉make命令需要处理哪些文件,以及如何处理;makefile文件u这些规则主要是描述:“目标文件”(不要和编译时产生的目标文件相混淆)是从哪些“相依文件”中产生的,以及用什么命令来执行这个产生过程。u目标文件不一定是最后的可执行文件,可以是任何一个中间文件,也可以是其他目标文件的依赖文件;u在此基础上,make命令对磁盘上的文件进行检查,如果目标文件的生成时间,或改动时间,比它的某个依赖文件还旧的话,make就执行相应的命令,更新目标文件;即:makefile涉及三方面内容:目标文件、相依文件和操作命令makefile文件示例:v假设:某个正在开发的程序包括prog.c和code.c两个C语言源文件,头文件有prog.h和code.h,且:vprog.c使用了prog.h和code.h两个头文件中声明的变量;v最后生成的可执行文件名为test;则,相应的makefile文件为:test:prog.ocode.ogccotestprog.ocode.oprog.o:prog.cprog.hcode.hgcccprog.ccode.o:code.ccode.hgccccode.cclean:rmf*.o目标文件目标文件位于冒号位于冒号左边左边相依文件位于冒号右边,相依文件位于冒号右边,通常是编译目标文件所通常是编译目标文件所需要的其他文件需要的其他文件1.根据相依文件,生根据相依文件,生成目标文件,所需成目标文件,所需执行的命令;执行的命令;2.每个命令占一行,每个命令占一行,且命令行的起始字且命令行的起始字符必须为符必须为TAB字符字符。目标文件可以是文件名,或者要执行的动作目标文件可以是文件名,或者要执行的动作:clean是常用的一种专用目标,用于删除所有的目是常用的一种专用目标,用于删除所有的目标模块。标模块。只要文件只要文件test的时间戳比文件的时间戳比文件prog.o或或code.o中的任何一个旧,下一行的中的任何一个旧,下一行的编译命令将会被执行。编译命令将会被执行。1.在检查文件prog.o和code.o的时间戳之前,make会在下面的行中寻找以prog.o和code.o为目标的规则;2.在第三行中找到了关于prog.o的规则,该文件的依赖文件是prog.c、prog.h和code.h;3.同样,make会在后面的规则行中继续查找这些依赖文件的规则,4.如果找不到,则开始检查这些依赖文件的时间戳,如果这些文件中任何一个的时间戳比prog.o的新,make将执行“gcc c prog.c o prog.o”命令,更新prog.o文件;make命令对上述规则的执行顺序:命令对上述规则的执行顺序:u以同样的方法,接下来对文件code.o做类似的检查,依赖文件是code.c和code.h。当make执行完所有这些套嵌的规则后,make将处理最顶层的test规则。u如果关于prog.o和code.o的两个规则中的任何一个被执行,至少其中一个.o目标文件就会比test新,那么就要执行test规则中的命令,将prog.o和code.o连接成目标文件test。通过以上的分析过程,可以看到通过以上的分析过程,可以看到通过以上的分析过程,可以看到通过以上的分析过程,可以看到makemakemakemake的优点的优点的优点的优点:u因为.o文件依赖.c源文件,源码文件里一个简单改变都会造成重新编译,并根据规则链依次由下到上执行编译过程,直到最终的可执行文件被重新连接;u例如,当改变一个头文件的时候,由于所有的依赖关系都在makefile里记录了,因此,make命令可以自动的重新编译所有那些因依赖这个头文件而改变了的源码文件,如果需要,再进行重新连接。vMake命令的工作过程如下:读入makefile文件;初始化文件中的变量;推导隐式规则,并分析所有规则;为所有的目标文件创建依赖关系链;根据依赖关系和时间数据,确定哪些目标文件要重新生成;执行相应的生成命令。在默认方式下,输入make命令,就可以调用它工作:在当前目录下寻找名字为makefile的文件,从第一个规则开始执行。6.3.2 使用变量vmakefile里的变量就像一个环境变量,一般使用大写宇母。变量的主要作用如下:保存文件名保存文件名 例如:使用一个变量来保存所有的目标文件名,则可以方便地加入新的目标文件而且不易出错。保存可执行命令名保存可执行命令名 如:编译命令,用一个变量来代替编译器名,那么只需要改变该变量的值。其他所有地方的命令名就都改变了。保存编译器的参数保存编译器的参数 很多源代码编译时,gcc需要很长的参数选项,在很多情况下,所有的编译命令使用一组相同的选项,如果把这组选项使用一个变量代表,那么可以把这个变量放在所有引用编译器的地方,当要改变选项的时候,只需改变一次这个变量的内容即可。1变量定义和引用v变量(又称做宏定义)一般均由大写字母和数字组成。v定义变量的一般格式是:=例如:OBJECT=x.o y.o z.o LIBES=-lmv引用make变量的方式与引用shell变量类似,把变量用圆括号括起来,并在前面加上“$”符号。例如:$(OBJECT)$(LIBES)现在利用变量把前面的makefile文件重写一遍:OBJS=prog.ocode.oCC=gcctest:$(OBJS)$(CC)otest$(OBJS)prog.o:prog.cprog.hcode.h$(CC)cprog.coprog.ocode.o:code.ccode.h$(CC)ccode.cocode.oclean:rmf*.otest:prog.ocode.ogccotestprog.ocode.oprog.o:prog.cprog.hcode.hgcccprog.coprog.ocode.o:code.ccode.hgccccode.cocode.oclean:rmf*.o本章作业 两个C语言源程序文件image.c和search.c共同完成一项任务,其中:image.c调用了search.h(保存在/temp目录中)中的函数dissort(),search.c使用了search.h中的变量。在Linux环境下用gcc命令进行编译,要求如下:1.生成可执行程序文件名为imgsch;2.加入供调试程序gdb使用的附加信息;请写出:1、gcc编译命令 2、makefile文件