GDB概述GDB是GNU开源组织发布的一个强大的UNIX下的程序调试工具。或许,各位比较喜欢那种图形界面方式的,像VC、BCB等IDE的调试,但如果你是在UNIX平台下做软件,你会发现GDB这个调试工具有比VC、BCB的图形化调试器更强大的功能。所谓“寸有所长,尺有所短”就是这个道理。一般来说,GDB主要帮忙你完成下面四个方面的功能:1、启动你的程序,可以按照你的自定义的要求随心所欲的运行程序。2、可让被调试的程序在你所指定的调置的断点处停住。(断点可以是条件表达式)3、当程序被停住时,可以检查此时你的程序中所发生的事。4、动态的改变你程序的执行环境。从上面看来,GDB和一般的调试工具没有什么两样,基本上也是完成这些功能,不过在细节上,你会发现GDB这个调试工具的强大,大家可能比较习惯了图形化的调试工具,但有时候,命令行的调试工具却有着图形化工具所不能完成的功能。让我们一一看来。一个调试示例源程序:tst.c1#includestdio.h23intfunc(intn)4{5intsum=0,i;6for(i=0;in;i++)7{8sum+=i;9}10returnsum;11}121314main()15{16inti;17longresult=0;18for(i=1;i=100;i++)19{20result+=i;21}2223printf(result[1-100]=%d\n,result);24printf(result[1-250]=%d\n,func(250));25}编译生成执行文件:(Linux下)hchen/testcc-gtst.c-otst使用GDB调试:hchen/testgdbtst----------启动GDBGNUgdb5.1.1Copyright2002FreeSoftwareFoundation,Inc.GDBisfreesoftware,coveredbytheGNUGeneralPublicLicense,andyouarewelcometochangeitand/ordistributecopiesofitundercertainconditions.Typeshowcopyingtoseetheconditions.ThereisabsolutelynowarrantyforGDB.Typeshowwarrantyfordetails.ThisGDBwasconfiguredasi386-suse-linux...(gdb)l--------------------l命令相当于list,从第一行开始例出原码。1#includestdio.h23intfunc(intn)4{5intsum=0,i;6for(i=0;in;i++)7{8sum+=i;9}10returnsum;(gdb)--------------------直接回车表示,重复上一次命令11}121314main()15{16inti;17longresult=0;18for(i=1;i=100;i++)19{20result+=i;(gdb)break16--------------------设置断点,在源程序第16行处。Breakpoint1at0x8048496:filetst.c,line16.(gdb)breakfunc--------------------设置断点,在函数func()入口处。Breakpoint2at0x8048456:filetst.c,line5.(gdb)infobreak--------------------查看断点信息。NumTypeDispEnbAddressWhat1breakpointkeepy0x08048496inmainattst.c:162breakpointkeepy0x08048456infuncattst.c:5(gdb)r---------------------运行程序,run命令简写Startingprogram:/home/hchen/test/tstBreakpoint1,main()attst.c:17----------在断点处停住。17longresult=0;(gdb)n---------------------单条语句执行,next命令简写。18for(i=1;i=100;i++)(gdb)n20result+=i;(gdb)n18for(i=1;i=100;i++)(gdb)n20result+=i;(gdb)c---------------------继续运行程序,continue命令简写。Continuing.result[1-100]=5050----------程序输出。Breakpoint2,func(n=250)attst.c:55intsum=0,i;(gdb)n6for(i=1;i=n;i++)(gdb)pi---------------------打印变量i的值,print命令简写。$1=134513808(gdb)n8sum+=i;(gdb)n6for(i=1;i=n;i++)(gdb)psum$2=1(gdb)n8sum+=i;(gdb)pi$3=2(gdb)n6for(i=1;i=n;i++)(gdb)psum$4=3(gdb)bt---------------------查看函数堆栈。#0func(n=250)attst.c:5#10x080484e4inmain()attst.c:24#20x400409edin__libc_start_main()from/lib/libc.so.6(gdb)finish---------------------退出函数。Runtillexitfrom#0func(n=250)attst.c:50x080484e4inmain()attst.c:2424printf(result[1-250]=%d\n,func(250));Valuereturnedis$6=31375(gdb)c---------------------继续运行。Continuing.result[1-250]=31375----------程序输出。Programexitedwithcode027.--------程序退出,调试结束。(gdb)q---------------------退出gdb。hchen/test好了,有了以上的感性认识,还是让我们来系统地认识一下gdb吧。使用GDB一般来说GDB主要调试的是C/C++的程序。要调试C/C++的程序,首先在编译时,我们必须要把调试信息加到可执行文件中。使用编译器(cc/gcc/g++)的-g参数可以做到这一点。如:cc-ghello.c-ohellog++-ghello.cpp-ohello如果没有-g,你将看不见程序的函数名、变量名,所代替的全是运行时的内存地址。当你用-g把调试信息加入之后,并成功编译目标代码以后,让我们来看看如何用gdb来调试他。启动GDB的方法有以下几种:1、gdbprogramprogram也就是你的执行文件,一般在当然目录下。2、gdbprogramcore用gdb同时调试一个运行程序和core文件,core是程序非法执行后coredump后产生的文件。3、gdbprogramPID如果你的程序是一个服务程序,那么你可以指定这个服务程序运行时的进程ID。gdb会自动attach上去,并调试他。program应该在PATH环境变量中搜索得到。GDB启动时,可以加上一些GDB的启动开关,详细的开关可以用gdb-help查看。我在下面只例举一些比较常用的参数:-symbolsfile-sfile从指定文件中读取符号表。-sefile从指定文件中读取符号表信息,并把他用在可执行文件中。-corefile-cfile调试时coredump的core文件。-directorydirectory-ddirectory加入一个源文件的搜索路径。默认搜索路径是环境变量中PATH所定义的路径。GDB的命令概貌启动gdb后,就你被带入gdb的调试环境中,就可以使用gdb的命令开始调试程序了,gdb的命令可以使用help命令来查看,如下所示:/home/hchengdbGNUgdb5.1.1Copyright2002FreeSoftwareFoundation,Inc.GDBisfreesoftware,coveredbytheGNUGeneralPublicLicense,andyouarewelcometochangeitand/ordistributecopiesofitundercertainconditions.Typeshowcopyingtoseetheconditions.ThereisabsolutelynowarrantyforGDB.Typeshowwarrantyfordetails.ThisGDBwasconfiguredasi386-suse-linux.(gdb)helpListofclassesofcommands:aliases--Aliasesofothercommandsbreakpoints--Makingprogramsatcertainpointsdata--Examiningdatafiles--Specifyingandexaminingfilesinternals--Maintenancecommandsobscure--Obscurefeaturesrunning--Runningtheprogramstack--Examiningthestackstatus--Statusinquiriessupport--Supportfacilitiestracepoints--Tracingofprogramexecutionwithoutspingtheprogramuser-defined--User-definedcommandsTypehelpfollowedbyaclassnameforalistofcommandsinthatclass.Typehelpfollowedbycommandnameforfulldocumentation.Commandnameabbreviationsareallowedifunambiguous.(gdb)gdb的命令很多,gdb把之分成许多个种类。help命令只是例出gdb的命令种类,如果要看种类中的命令,可以使用helpclass命令,如:helpbreakpoints,查看设置断点的所有命令。也可以直接helpcommand来查看命令的帮助。gdb中,输入命令时,可以不用打全命令,只用打命令的前几个字符就可以了,当然,命令的前几个字符应该要标志着一个唯一的命令,在Linux下,你可以敲击两次TAB键来补齐命令的全称,如果有重复的,那么gdb会把其例出来。示例一:在进入函数func时,设置一个断点。可以敲入breakfunc,或是直接就是bfunc(gdb)bfuncBreakpoint1at0x8048458:filehello.c,line10.示例二:敲入b按两次TAB键,你会看到所有b打头的命令:(gdb)bbacktracebreakbt(gdb)示例三:只记得函数的前缀,可以这样:(gdb)bmake_按TAB键(再按下一次TAB键,你会看到:)make_a_section_from_filemake_environmake_abs_sectionmake_function_typemake_blockvectormake_pointer_typemake_cleanupmake_reference_typemake_commandmake_symbol_completion_list(gdb