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函数的入口处设置断点;breakfilename: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命令,可以显示某个变量的类型;

要想了解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:

#ifndef _ADD_H_
#define _ADD_H_

int CalAdd(int a, int b);
float CalAdd(float a, float  b, float c);

#endif //_ADD_H_

subtract.h:

#ifndef _SUBTRACT_H_
#define _SUBTRACT_H_

int CalSubtract(int a, int b);
float CalSubtract(float a, float b);

#endif //_SUBTRACT_H_

add.cpp:

#include "../../include/add/add.h"

int CalAdd(int a, int b)
{
	int tmp1 = a / 2;

	if (b % 2 == 1) {
		b = b * 2 + 3;
	} else {
		b = b + 5;
	}

	int tmp2 = -1;

	tmp1 = a + tmp1 + b + tmp2;

	return tmp1;
}

float CalAdd(float a, float b, float c)
{
	float tmp1 = a - b;

	for (int i = 0; i < 10; i ++) {
		float tmp2 = i * 0.5;

		tmp1 += tmp2;
	}

	float tmp2 = a + c - b - tmp1;

	return tmp2;
}

subtract.cpp:

#include "../../include/subtract/subtract.h"

int CalSubtract(int a, int b)
{
	int tmp1 = 5;

	while (tmp1) {
		int tmp2 = 100;

		a += tmp2 / 2;

		-- tmp1;
	}

	int  tmp3 = a - b;

	return tmp3;
}

float CalSubtract(float a, float b)
{
	float tmp1 = -111;

	for (int i = 0; i < 100; i ++) {
		float tmp2 = -199;

		a -= b;

		a += tmp2;
	}

	float tmp3 = a - b;

	return tmp3;
}

test.cpp:

#include <iostream>
#include <stdlib.h>
#include "../../include/add/add.h"
#include "../../include/subtract/subtract.h"

using namespace std;

int main(int argc, char* argv[])
{
	cout<<"integer test, input integer:"<<endl;

	int a = 5, b = 10;

	if (argc > 1)
	  a = atoi(argv[1]);

	if (argc > 2)
	  b = atoi(argv[2]);

	int ret = 0;

	cout<<"start add:"<<endl;
	ret = CalAdd(a, b);
	cout<<"integer add :"<<ret<<endl;

	cout<<"start subtract:"<<endl;
	ret = CalSubtract(b, a);
	cout<<"integer subtract:"<<ret<<endl;

	cout<<"ok!!"<<endl;

	return 0;
}

打开终端,定位为到/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

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。