GDB调试变量、内存与寄存器查看与修改 _

变量查看与修改

在使用 GDB 调试程序时,除了控制程序的运行流程以外,最常见的操作就是查看变量的值、查看参数的值,以及在调试过程中临时修改某些变量的值。通过这些命令,可以观察程序运行时的数据变化,从而判断程序逻辑是否符合预期。

1.1 查看函数参数

使用info args(i args)命令可以查看当前函数传入参数的值,包括main函数的argvargc参数。

i args

例如,当前程序停在某个函数内部:

void test(int a, int b) { int sum = a + b; }

在断点处执行:

i args

就可以看见传入的ab变量的值

除了使用i args,也可以直接使用print(p)命令打印某个参数的值:

p a p b

1.2 查看变量

1、查看普通变量

在 GDB 中,可以使用print命令查看变量的值,print可以简写为p

p [变量名]

例如:

p num p ch

默认情况下,p命令会按照变量本身的类型打印结果。如果想按照不同格式查看变量,可以在p后面加格式控制。

常见格式如下:

命令含义
p/d var按十进制打印
p/x var按十六进制打印
p/t var按二进制打印
p/o var按八进制打印
p/c var按字符打印
p/s var按字符串打印

例如:

p/d num p/x num p/t num p/c ch

如果有如下代码:

int num = 65; char ch = 'A';

执行:

p/d num p/x num p/c num

可能得到:

$1 = 65 $2 = 0x41 $3 = 65 'A'

这样可以方便地从不同角度查看变量的值。


2、查看变量类型

如果想查看变量的类型,可以使用如下命令:

ptype [变量名]

ptype显示的信息详细,适合查看结构体、指针、数组等复杂类型。

例如,有如下结构体定义:

struct Student { char name[32]; int age; }; struct Student stu = {"Tom", 18}; struct Student *pstu = &stu;

在 GDB 中可以执行:

ptype stu

可能得到:

type = struct Student { char name[32]; int age; }

这说明变量stu的类型是struct Student,并且 GDB 会把结构体中的成员也显示出来。

如果查看结构体指针变量:

ptype pstu

可能得到:

type = struct Student *

这说明pstu是一个指向struct Student类型的指针。

如果想查看指针指向的对象类型,也可以对指针解引用后再查看:

ptype *pstu

可能得到:

type = struct Student { char name[32]; int age; }

在调试结构体、链表、树、数组指针等复杂数据结构时,ptype非常有用,可以帮助我们快速确认变量的数据类型以及内部成员组成。

whatis命令可以用来查看结构体,类,派生类,普通函数以及类成员函数
ptype命令可以更加详细的查看这些内容,如果显示一个类,ptype会把类的成员变量和成员函数都显示出来i variables`命令用来查看全局变量或者静态变量,比如我们看见一个变量使用,但是不知道该变量是在哪定义的,就可以使用这个命令

ptype /m //只显示成员变量,不显示成员函数 ptype /t //不现实typedef的内容 ptype /o //显示结构体或者类的内存构造以及大小

例如如果我们对结构体占用内存大小进行优化,可以使用ptype命令对结构体进行查看,查看他对内存空间的利用率

(gdb) ptype /o node /* offset | size */ type = struct NODE { /* 0 | 4 */ int ID; /* 4 | 4 */ int age; /* 8 | 1 */ char gender; /* XXX 7-byte hole */ /* 16 | 8 */ NODE *next; /* 24 | 4 */ int test; /* 28 | 1 */ char c; /* XXX 3-byte padding */ /* total size (bytes): 32 */ }

可以看见对于上面的结构体,所有成员直接加起来才22个字节,但是使用不合理的排布之后,会用32个字节才能存下,调整结构体之后,可以优化内存布局到24个字节

(gdb) ptype /o node2 /* offset | size */ type = struct NODE2 { /* 0 | 4 */ int ID; /* 4 | 4 */ int age; /* 8 | 8 */ NODE *next; /* 16 | 4 */ int test; /* 20 | 1 */ char gender; /* 21 | 1 */ char c; /* XXX 2-byte padding */ /* total size (bytes): 24 */ }