GDB 调试之ptype、set variable

  • Post author:
  • Post category:其他


今天在公司的时候,排查一个问题,创建l3 lif 失败,查看各种日志发现是用key去创建的 lif失败了,日志里指示key为空,导致的创建失败。原因为一个结构体比基线的多了一些东西,导致版本不对,既而计算出来的key不对。调试过程中用到了 ptype 打印结构体变量的类型,结果看基线分支代码却对应不上,比基线代码的结构体多了一个 union 变量。以下只作一下练习:

#include <map>
#include <string>
#include <stdlib.h>
#include <stdio.h>

enum
{
    TYPE_ID_0,
    TYPE_ID_1,
    TYPE_ID_2,
    TYPE_ID_3,
    TYPE_ID_4,
    TYPE_ID_5,
    TYPE_ID_6,
    TYPE_ID_MAX
};
typedef struct mapData
{
    int id;
    char *p;
    std::string key;
}MapData;

std::map<std::string, MapData> g_mapData[TYPE_ID_MAX];

int changeMap(int type);
int main()
{
    changeMap(2);

    for(int i = 0 ; i < TYPE_ID_MAX; i++)
    {
        auto data = g_mapData[i];
        //for(auto it : data)
        for(auto ite = data.begin(); ite != data.end(); ite++)
        {
            //printf("i = %d, string: %s, id = %d, key = %s\n", i, it.first.c_str(), it.second.id, it.second.key.c_str());
            printf("i = %d, string: %s, id = %d, key = %s\n", i, ite->first.c_str(), ite->second.id, ite->second.key.c_str());
        }
    }

    return 0;
}

int changeMap(int type)
{
    int a = 100;
    auto &data = g_mapData[type];
    char test[] = "test";
    MapData newData = {20, test, "2000"};
    data["1"] = newData;
    data.insert(std::pair<std::string, MapData>("2", newData));
    return 0;
}

用GDB调试,打一个断点:

我们可以看到如果是结构体类型,则会打印出来结构体的成员,如果是map类型会打印出什么呢?

如果要改变变量的值,可以使用命令 set variable xx=xx,如:

如果是结构体类型该怎么赋值呢?先看一下最简单的,结构里就一个基本的类型:

#include <map>
#include <string>
#include <stdlib.h>
#include <stdio.h>

enum
{
    TYPE_ID_0,
    TYPE_ID_1,
    TYPE_ID_2,
    TYPE_ID_3,
    TYPE_ID_4,
    TYPE_ID_5,
    TYPE_ID_6,
    TYPE_ID_MAX
};
typedef struct mapData
{
    int id;
}MapData;

std::map<std::string, MapData> g_mapData[TYPE_ID_MAX];

int changeMap(int type);
int main()
{
    changeMap(2);

    for(int i = 0 ; i < TYPE_ID_MAX; i++)
    {
        auto data = g_mapData[i];
        //for(auto it : data)
        for(auto ite = data.begin(); ite != data.end(); ite++)
        {
            //printf("i = %d, string: %s, id = %d, key = %s\n", i, it.first.c_str(), it.second.id, it.second.key.c_str());
            //printf("i = %d, string: %s, id = %d, key = %s\n", i, ite->first.c_str(), ite->second.id, ite->second.key.c_str());
        }
    }

    return 0;
}

int changeMap(int type)
{
    int a = 100;
    auto &data = g_mapData[type];
    char test[] = "test";
    MapData newData = {20};
    data["1"] = newData;
    data.insert(std::pair<std::string, MapData>("2", newData));
    return 0;
}

打个断点,查看一下结构体的值,然后再修改一下,看结果:


如果是复杂的结构体呢?如:

typedef struct mapData
{
    int id;
    char *p;
    std::string key;
}MapData;

结构体里有指针,有std::string 类型,这时候该怎么赋值呢?

结果有点不一样,即使第二个成员是指针,我们临时给它赋了一个值,也是正常的,但std::string却显示为空,为何?我们看一下std::string 的源码:


string 保存的是一个指针,指针指向的才是真正的数据,那我们可以看一下这个指针指向的地址是不是保存着我们给它赋的值 “aaa” :

首先打印出key的地址,再根据它的地址看内存里保存了哪些值,其中 0x00607060 就是std::string 私有成员 _M_dataplus的值,因为它是一个指针,指向的是string 的内容。但为何在GDB里直接打印结构体却显示为空呢?有知道的大神指点一下。



版权声明:本文为tianyexing2008原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。