GDB概述GDB是GNU开源组织发布的一个强大的UNIX下的程序调试工具。或许,各位比较喜欢那种图形界面方式的,像VC、BCB等IDE的调试,但如果你是在UNIX平台下做软件,你会发现GDB这个调试工具有比VC、BCB的图形化调试器更强大的功能。所谓“寸有所长,尺有所短”就是这个道理。一般来说,GDB主要帮忙你完成下面四个方面的功能:1.启动你的程序,可以按照你的自定义的要求随心所欲的运行程序。2.可让被调试的程序在你所指定的调置的断点处停住。(断点可以是条件表达式)3.当程序被停住时,可以检查此时你的程序中所发生的事。4.动态的改变你程序的执行环境。从上面看来,GDB和一般的调试工具没有什么两样,基本上也是完成这些功能,不过在细节上,你会发现GDB这个调试工具的强大,大家可能比较习惯了图形化的调试工具,但有时候,命令行的调试工具却有着图形化工具所不能完成的功能。让我们一一看来。[编辑]一个调试示例源程序:tst.c#includestdio.hintfunc(intn){intsum=0,i;for(i=1;i=n;i++){sum+=i;}returnsum;}intmain(){inti;longresult=0;for(i=1;i=100;i++){result+=i;}printf(result[1-100]=%ld\n,result);printf(result[1-250]=%d\n,func(250));return0;}编译生成执行文件:$gcc-g-Walltst.c-otst使用GDB调试:$gdbtst----------启动GDBGNUgdb6.7.1-debianCopyright(C)2007FreeSoftwareFoundation,Inc.LicenseGPLv3+:GNUGPLversion3orlater:youarefreetochangeandredistributeit.ThereisNOWARRANTY,totheextentpermittedbylaw.Typeshowcopyingandshowwarrantyfordetails.ThisGDBwasconfiguredasi486-linux-gnu...Usinghostlibthread_dblibrary/lib/tls/i686/cmov/libthread_db.so.1.(gdb)l1--------------------l命令相当于list,从第一行开始列出源码。1#includestdio.h23intfunc(intn)4{5intsum=0,i;6for(i=1;i=n;i++)7{8sum+=i;9}10returnsum;(gdb)--------------------直接回车表示,重复上一次命令11}121314intmain()15{16inti;17longresult=0;18for(i=1;i=100;i++)19{20result+=i;(gdb)break16--------------------设置断点,在源程序第16行处。Breakpoint1at0x80483b2:filetst.c,line16.(gdb)breakfunc--------------------设置断点,在函数func()入口处。Breakpoint2at0x804837a:filetst.c,line5.(gdb)infobreak--------------------查看断点信息。NumTypeDispEnbAddressWhat1breakpointkeepy0x080483b2inmainattst.c:162breakpointkeepy0x0804837ainfuncattst.c:5(gdb)r---------------------运行程序,run命令简写Startingprogram:/home/dbzhang/tstBreakpoint1,main()attst.c:1717longresult=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=-1074568236(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:6#10x080483f1inmain()attst.c:24(gdb)finish---------------------退出函数。Runtillexitfrom#0func(n=250)attst.c:60x080483f1inmain()attst.c:2424printf(result[1-250]=%d\n,func(250));Valuereturnedis$5=31375(gdb)c---------------------继续运行。Continuing.result[1-250]=31375----------程序输出。Programexitednormally.--------程序退出,调试结束。(gdb)q---------------------退出gdb。好了,有了以上的感性认识,还是让我们来系统地认识一下gdb吧。[编辑]使用GDB一般来说GDB主要调试的是C/C++的程序。要调试C/C++的程序,首先在编译时,我们必须要把调试信息加到可执行文件中。使用编译器(cc/gcc/g++)的-g参数可以做到这一点。如:$gcc-g-Wallhello.c-ohello$g++-g-Wallhello.cpp-ohello如果没有-g,你将看不见程序的函数名、变量名,所代替的全是运行时的内存地址。当你用-g把调试信息加入之后,并成功编译目标代码以后,让我们来看看如何用gdb来调试他。启动GDB的方法有以下几种:gdbprogramprogram也就是你的执行文件,一般在当前目录下。gdbprogramcore用gdb同时调试一个运行程序和core文件,core是程序非法执行后coredump后产生的文件。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命令来查看,如下所示:$gdbGNUgdb6.7.1-debianCopyright(C)2007FreeSoftwareFoundation,Inc.LicenseGPLv3+:GNUGPLversion3orlater:youarefreetochangeandredistributeit.ThereisNOWARRANTY,totheextentpermittedbylaw.Typeshowcopyingandshowwarrantyfordetails.ThisGDBwasconfiguredasi486-linux-gnu.(gdb)helpListofclassesofcommands:aliases--Aliasesofothercommandsbreakpoints--Makingprogramstopatcertainpointsdata--Examiningdatafiles--Specifyingandexaminingfilesinternals--Maintenancecommandsobscure--Obscurefeaturesrunning--Runningtheprogramstack--Examiningthestackstatus--Statusinquiriessupport--Supportfacilitiestracepoints--Tracingofprogramexecutionwithoutstoppingtheprogramuser-defined--User-definedcommandsTypehelpfollowedbyaclassnameforalistofcommandsinthatclass.Typehelpallforthelistofallcommands.Typehelpfollowedbycommandnameforfulldocumentation.Typeaproposwordtosearchforcommandsrelatedtoword.Commandnameabbreviationsareallowedifunambiguous.(gdb)gdb的命令很多,gdb把之分成许多个种类。help命令只是例出gdb的命令种类,如果要看种类中的命令,可以使用helpclass命令,如:helpbreakpoints,查看设置断点的所有命令。也可以直接helpcommand来查看命令的帮助。gdb中,输入命令时,可以不用打全命令,只用打命令的前几个字符就可以了,当然,命令的前几个字符应该要标志着一个唯一的命令,在Linux下,你可以敲击两次TAB键来补齐命令的全称,如果有重复的,那么gdb会把其例出来。示例一:在进入函数func时,设置一个断点。可以敲入breakfunc,或是直接就是bfunc(gdb)bfuncBreakpoint1at0x804837a:filetst.c,line5.示例二:敲入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_