当前位置:   article > 正文

Linux下gdb调试工具的使用_linux下调试器gdb的使用

linux下调试器gdb的使用

gdb是GNU开源组织发布的一个强大的Linux下的程序调试工具。

gdb主要完成四个方面的功能:(1)、启动你的程序,可以按照你的自定义的要求随心所欲的运行程序;(2)、可让被调试的程序在你所指定的调试的断点处停住(断点可以是条件表达式);(3)、当程序被停住时,可以检查此时你的程序中所发生的事;(4)、动态的改变你程序的执行环境。

要想运行准备调试的程序,可使用run命令,在它后面可以跟随发给该程序的任何参数,包括标准输入和标准输出说明符(<;和>;)和shell通配符(*、?、[、])在内。如果你使用不带参数的run命令,gdb就再次使用你给予前一条run命令的参数。利用set  args命令就可以修改发送给程序的参数,而使用show  args命令就可以查看其缺省参数的列表。

一般来说,gdb主要调试的是c/c++的程序,也可以调试Ada、Objective-C、Pascal及其它语言。要调试c/c++的程序,首先在编译时,我们必须要把调试信息加到可执行文件中。使用编译器(cc/gcc/g++)的-g(-ggdb)参数可以做到这一点。如果没有-g,你将看不见程序的函数名、变量名,所代替的全是运行时的内存地址。

启动gdb的方法:(1)、gdb  <program> :program就是执行文件,一般在当前目录下;(2)、gdb  <program>  core:用gdb同时调试一个运行程序和core文件,core是程序非法执行后core  dump后产生的文件;(3)、gdb  <program>  <PID> :如果是一个服务程序,你可以指定这个服务程序运行时的进程ID。gdb会自动attach上去,并调试它。

gdb中,输入命令时,可以不用打全命令,只用打命令的前几个字符就可以了,当然,命令的前几个字符应该要标志着一个唯一的命令,在Linux下,可以敲击两次Tab键来补齐命令的全称,如果有重复的,那么gdb会把相关的全例出来。

gdb调试中常用的命令:

(1)、l或list命令,查看源码;

(2)、r或run命令,运行程序,会在有断点的地方停住,如果是有输入参数的,则为run  argv[1]  argv[2];

(3)、break  n,在源程序第n行处设置断点;break  fun,在函数fun()入口处设置断点;break  filename:function ,在源文件filename的function函数的入口处设置断点;break filename:linenum ,在源文件filename的第linenum行处设置断点;break  if<condition>, 在条件成立时停住;filename可以是相对路径,即你不仅可以在当前原文件中设置断点,也可以在该程序中用到的其它目录中的原文件中设置断点;

(4)、info命令,查看指定命令信息,如info  break会显示所有设置的断点信息;

(5)、n或next命令,单条语句执行,如果有函数调用,它不会进入该函数;

(6)、c或continue命令,继续运行程序;

(7)、p或print 变量名var,打印变量var的值。print是gdb的一个功能很强的命令,利用它可以显示被调试的语言中任何有效的表达式。表达式除了包含你程序中的变量外,还可以包含以下内容:A、对程序中函数的调用;B、数据结构和其它复杂对象;

(8)、bt或backtrace命令,查看函数调用栈的所有信息,当程序执行异常时,可通过此命令查看程序的调用过程;

(9)、finish命令,运行程序,直到当前函数完成返回;

(10)、q或quit命令,退出gdb;

(11)、set命令,可指定运行时变量名的值;

(12)、watch  <expr>,为表达式(变量)expr设置一个观察点,一旦表达式的值有变化时,马上停住程序;rwatch <expr>,当表达式(变量)expr被读时,停住程序;awatch  <expr>,当表达式(变量)的值被读或被写时,停住程序;

(13)、clear命令,清除所有的已定义的停止点,如断点等;clear <filename:function>,清除所有设置在函数上的停止点;clear <filename:linenum>,清除所有设置在指定行上的停止点;

(14)、step命令,单步跟踪,如果有函数调用,它会进入该函数,如果想退出该函数返回到它的调用函数中,可以使用finish命令;

(15)、jump命令,可以修改程序的执行顺序,可以让程序执行随意跳跃;

(16)、whatis命令,可以显示某个变量的类型;

(17)、up和down命令:遍历函数堆栈,up是将调用栈上移一个函数调用,down是将调用栈下移一个函数调用;

(18)、where命令:查看逆序的调用堆栈信息;

要想了解gdb的更详细的用法,可以从http://www.gnu.org/software/gdb/documentation/下载gdb  manuals。

以下以一个测试程序为例,说明其中的一些命令的具体用法,此例程的组织结构为:gdbtest为总目录,下面有include/demo/src三个目录,(1)、include/add/add.h, include/subtract/subtract.h;(2)、src/add/add.cpp,src/subtract/subtract.cpp;(3)、demo/test/test.cpp。

每个文件的详细内容为:

add.h:

  1. #ifndef _ADD_H_
  2. #define _ADD_H_
  3. int CalAdd(int a, int b);
  4. float CalAdd(float a, float b, float c);
  5. #endif //_ADD_H_

subtract.h:

  1. #ifndef _SUBTRACT_H_
  2. #define _SUBTRACT_H_
  3. int CalSubtract(int a, int b);
  4. float CalSubtract(float a, float b);
  5. #endif //_SUBTRACT_H_

add.cpp:

  1. #include "../../include/add/add.h"
  2. int CalAdd(int a, int b)
  3. {
  4. int tmp1 = a / 2;
  5. if (b % 2 == 1) {
  6. b = b * 2 + 3;
  7. } else {
  8. b = b + 5;
  9. }
  10. int tmp2 = -1;
  11. tmp1 = a + tmp1 + b + tmp2;
  12. return tmp1;
  13. }
  14. float CalAdd(float a, float b, float c)
  15. {
  16. float tmp1 = a - b;
  17. for (int i = 0; i < 10; i ++) {
  18. float tmp2 = i * 0.5;
  19. tmp1 += tmp2;
  20. }
  21. float tmp2 = a + c - b - tmp1;
  22. return tmp2;
  23. }

subtract.cpp:

  1. #include "../../include/subtract/subtract.h"
  2. int CalSubtract(int a, int b)
  3. {
  4. int tmp1 = 5;
  5. while (tmp1) {
  6. int tmp2 = 100;
  7. a += tmp2 / 2;
  8. -- tmp1;
  9. }
  10. int tmp3 = a - b;
  11. return tmp3;
  12. }
  13. float CalSubtract(float a, float b)
  14. {
  15. float tmp1 = -111;
  16. for (int i = 0; i < 100; i ++) {
  17. float tmp2 = -199;
  18. a -= b;
  19. a += tmp2;
  20. }
  21. float tmp3 = a - b;
  22. return tmp3;
  23. }

test.cpp:

  1. #include <iostream>
  2. #include <stdlib.h>
  3. #include "../../include/add/add.h"
  4. #include "../../include/subtract/subtract.h"
  5. using namespace std;
  6. int main(int argc, char* argv[])
  7. {
  8. cout<<"integer test, input integer:"<<endl;
  9. int a = 5, b = 10;
  10. if (argc > 1)
  11. a = atoi(argv[1]);
  12. if (argc > 2)
  13. b = atoi(argv[2]);
  14. int ret = 0;
  15. cout<<"start add:"<<endl;
  16. ret = CalAdd(a, b);
  17. cout<<"integer add :"<<ret<<endl;
  18. cout<<"start subtract:"<<endl;
  19. ret = CalSubtract(b, a);
  20. cout<<"integer subtract:"<<ret<<endl;
  21. cout<<"ok!!"<<endl;
  22. return 0;
  23. }

打开终端,定位为到/demo/test/目录下,依次输入:

g++  -c  -g  test.cpp  ../../src/add/add.cpp  ../../src/subtract/subtract.cpp
g++  -o  test  test.o  add.o  subtract.o

会在test目录下生成test.o, add.o, subtract.o和test执行文件。

下面开始调试test:

(1)、输入gdb test,会显示:

(2)、先用l或list命令,查看当前原文件源码,每次会显示10行,每次显示行数也可以设置,如果想继续查看后面的代码,可以直接按Enter键,表示继续执行上次输入的命令:

(3)、为了不让程序一次性执行完,可以先设置一个断点,后面根据需要可以随时再增加断点,输入break  22,在当前文件test.cpp的第22行设置断点,输入info break查看此断点信息:

(4)、使用run命令开始正式调试test,输入run 10  20:

(5)、使用n或next命令,单步调试,接着输入step命令进入到CalAdd函数内部:

(6)、多设置几个断点:break  ../../src/add/add.cpp:13,break ../../subtract/subtract.cpp:5, break 26, 输入info break查看:

(7)、输入finish命令,从CalAdd函数内部退出,使用whatis命令,查看变量ret类型,输入print ret查看变量值:

(8)、使用continue命令,执行到下一个断点设置处:

(9)、使用clear命令,清除指定的断点:

(10)、如果想从头再次重新调试程序,可接着输入run 30  40,之前设的断点继续有效,如退出gdb调试,直接输入q或quit即可:

参考文献:

1. http://www.gnu.org/software/gdb/

2. http://blog.csdn.net/haoel/article/category/9197

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Monodyee/article/detail/288074
推荐阅读
相关标签
  

闽ICP备14008679号