您好,登录后才能下订单哦!
GDB调试C++类
Linux上调试常用的工具就是gdb了。借助学习C++虚函数表和内存布局的机会顺便学习下gdb常规调试技巧。
一,测试用例
1,C++头文件(szyu_test_gdb.h)
/******************************
*
* Author : szyu
*
* Date : 2016.10.25
*
********************************/
#ifndef __SZYU_GDB__
#define __SZYU_GDB__
#include <iostream>
class Base
{
public:
    Base() { };
    Base( int v ) : non_static_member1( v ) { };
    virtual ~Base() { };
public:
    void
    non_static_func1()
    {   
        std::cout << "In non_static_func1()" << std::endl;
    }   
    static void
    static_func1()
    {
        std::cout << "In static_func1()" << std::endl;
    }
    virtual void
    virtual_func1()
    {
        std::cout << "In virtual_func1()" << std::endl;
    }
private:
    int non_static_member1;
    static int static_member1;
};
int Base::static_member1 = 99;
#endif2,C++测试用例(szyu_test_gdb.cpp)
/************************
*
* Author : szyu
*
* Date : 2016.10.25
*
******************************/
#include "szyu_test_gdb.h"
void
test1()
{
    /* 静态函数访问 */
    Base::static_func1();
    /* 创建对象 */
    Base bb( 57 );
    /* 非静态函数访问 */
    bb.non_static_func1();
    /* 虚函数访问 */
    bb.virtual_func1();
}
int
main( int argc, char *argv[] )
{
    test1();
    return 0;
}二,调试
1,gdb调试前需编译生成可执行文件,并且需把调试信息加到可执行文件中。-g参数可以做到这点。使用方法为:g++ -g szyu_test_gdb.cpp(默认生成a.out可执行文件)
2,启动gdb调试:gdb a.out
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227329.jpg)
3,设置断点,可获取运行时的堆栈信息。
分别对以下位置设置了断点:
1)构造函数:Base( int v )
2)虚析构函数:virtual ~Base()
3)静态函数调用:bb.static_func1()
4)非静态函数调用:bb.non_static_func1()
5)虚函数调用:bb.virtual_func1()
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227330.jpg)
4,运行程序,遇到第一个断点:静态函数调用。单步跟踪s(step)命令进入函数内部,并打印地址如下:
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227331.jpg)
5,c(continue)恢复程序运行,接下来程序停在第二断点处,即Base(int v)构造函数处。打印Base类对象bb如下:
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227332.jpg)
由图可以类对象的地址为:0x7fffffffe320
对对象首地址进行解引用得到地址:0x400cb0即为虚函数表地址,故可知在该编译器中的虚函数表位于对象实例的最前端。
从打印的Base类对象可知,此时虚函数表已创建,且虚函数表地址为0x400cb0。对于静态成员和非静态成员都已初始化好了。此时获取虚函数表中的内容如下所示:
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227333.jpg)
由于虚函数表是二级指针。所以使用void **转换。再使用解引用运算符变成一级指针。
其中还涉及到set print array,@和/a三个知识点:
1)默认数组显示是关闭状态的(即打印数组时,每个元素则以逗号分隔)。打开数组显示状态后,每个元素占一行打印。
2)p /a 打印语句中a只是参数选项之一,常见该参数如下:
    x  按十六进制格式显示变量。
    d  按十进制格式显示变量。
    u  按十六进制格式显示无符号整型。
    o  按八进制格式显示变量。
    t  按二进制格式显示变量。 
    a  按十六进制格式显示变量。
    c  按字符格式显示变量。
    f  按浮点数格式显示变量。
3)“@”的左边是第一个内存的地址的值,“@”的右边则你你想查看内存的长度。上图中的4代表打印出四段内存长度。此处由于虚函数表中总共存了三个虚函数内存段地址,故最后一个值是随机数。
为了支持RTTI(Run Time Type Identification,运行时类型识别),在虚函数表前存放了type_info指针。
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227334.jpg)
而静态成员变量和非静态成员变量获取如下:
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227336.jpg)
6,c(continue)继续运行,接下来程序停留在第三个断点处,即非静态方法调用。
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227337.jpg)
7,c(continue)继续运行,接下来程序停留在第四个断点处,即虚函数调用。
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227338.jpg)
由打印出来的虚函数地址可知:该地址与虚函数表中存的地址一致。
8,c(continue)继续运行,接下来程序停留在第五个断点处,即虚析构函数。
![[C++]GDB调试C++类](https://cache.yisu.com/upload/information/20200311/59/227339.jpg)
通过info line查看地址与虚函数表一致。
本文gdb调试命令主要参考:http://blog.csdn.net/haoel/article/details/2879
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。