【GDB】调试Jsoncpp源码

前言:

起初在写jsoncpp样例的时候,写出了一个这样的悬垂指针的bug,代码如下:

复制代码
int main()
{
    Json::Value root;
    root["name"] = "zhangsan";
    root["age"] = 18;
    root["sex"] = "mele";
    root["score"].append(90);
    root["score"].append(80);
    root["score"].append(70);

    // 实例化
    Json::StreamWriterBuilder builder;
    builder["indentation"] = "";
    std::unique_ptr<Json::StreamWriter> writer(builder.newStreamWriter());
    // 调用write函数
    std::stringstream ss;
    int ret = writer->write(root, &ss);
    if(ret != 0)
    {
        cout << "write failed" << endl;
    }
    else
    {
        cout << "write success" << endl;
        cout << ss.str() << endl;
    }
    // // 反序列化
    Json::CharReaderBuilder readerBuilder;
    std::unique_ptr<Json::CharReader> reader(readerBuilder.newCharReader());
    Json::Value root2;
    std::string errors;
    bool ret1;
    ret1 = reader->parse(ss.str().c_str(), ss.str().c_str() + ss.str().size(), &root2, &errors);
    
    if(!ret1)
    {
        cout << "parse failed" << endl;
    }
    else
    {
        cout << "parse success" << endl;
        cout << root2["name"].asString() << endl;
        cout << root2["age"].asInt() << endl;
        cout << root2["sex"].asString() << endl;
        for(int i = 0; i < root2["score"].size(); i++)
        {
            cout << root2["score"][i].asInt() << endl;
        }
    }
    return 0;
}

我发现ret1返回的是false而前面的代码又没问题,然后我想着GDB调试一下吧看看parse内部咋回事(当时我Ctrl 点击这个reader->parse发现进入的一直是头文件,那时的我以为GDB调试一下就能进入parse的源码)后面就衍生出了诸多问题。。。。。。。。

先说如何编译 然后直接GDB调试:

1.先把源码下载到本地

复制代码
git clone https://github.com/open-source-parsers/jsoncpp.git

2.进入jsoncpp源码创建目录并开始编译

复制代码
// 创建目录
mkdir build
// 进入目录 
cd build
//因为源码是 set(CMAKE_BUILD_TYPE Release CACHE STRING 是release版的所以我们直接命令行输入DEBUG版的
cmake -DCMAKE_BUILD_TYPE=Debug ..
// 编译
make

3.链接第三方库

这里我展示一下我的目录结构

复制代码
// 这是我链接的方式
g++ -o main main.cc -I./jsoncpp/include -L./jsoncpp/build/lib -ljsoncpp -g -std=c++11 -Wl,-rpath=./jsoncpp/build/lib

// 查看一下链接信息
ldd main

OK已经连接第三方库了

复制代码
gdb main

// 先,添加目录到源码搜索路径
(gdb) dir /home/dev/workspace/mystudy/jsoncpp/jsoncpp/src/lib_json
// 打断点
(gdb) b 64
(gdb) b json_reader.cpp:1855
// 这里我们info b一下查看断点信息,这里Address<PENDING>不用管。后面我们就正常的调试就行了
(gdb) info b
Num     Type           Disp Enb Address            What
1       breakpoint     keep y   0x0000000000004a12 in main() at main.cc:64
2       breakpoint     keep y   <PENDING>          json_reader.cpp:1855

后面就使用正常的操作直接r 启动程序 n下一步 s进入函数内部这种的基础操作了。


最后得出的结论是调用ss.str()创建了两个不同的临时字符串对象,它们的指针指向不同的内存区域,导致parse函数收到了无效的指针范围。

正确写法

复制代码
string str = ss.str();
ret1 = reader->parse(str.c_str(), str.c_str() + str.size(), &root2, &errors);

可是我打印的地址都一样啊。

复制代码
std::cout << (void*)ss.str().c_str() << std::endl;  // 0x1234
std::cout << (void*)ss.str().c_str() << std::endl;  // 0x1234 

但这只是巧合或编译器优化,不能依赖!原因:1.第一个临时对象销毁后,其内存可能被立即重用。2.编译器可能进行优化,但不能保证。

> - < 哭了,原来编译器优化也增加学习成本。

举个栗子

复制代码
#include <cstring>
#include <iostream>
#include <sstream>

void test_pointers(const char *start, const char *end) {
  std::cout << "start: " << (void *)start << ", end: " << (void *)end
            << std::endl;

  // 检查是否在同一个内存块
  size_t block_size = (end > start) ? (end - start) : 0;
  std::cout << "Block size calculated: " << block_size << std::endl;

  // 尝试读取(可能崩溃或读取错误数据)
  if (start && end && end > start) {
    std::cout << "First few chars at start: ";
    for (int i = 0; (start + i) < end; i++) {
      std::cout << start[i];
    }
    std::cout << std::endl;
  }
}

int main() {
  std::stringstream ss;
  ss << R"({"name":"zhangsan"})";

  // 错误方式
  std::cout << "=== 错误方式(临时对象)===" << std::endl;
  const char *start_bad = ss.str().c_str();
  const char *end_bad = ss.str().c_str() + ss.str().size();
  test_pointers(start_bad, end_bad);

  std::cout << "\n=== 正确方式(单个对象)===" << std::endl;
  // 正确方式
  std::string jsonStr = ss.str();
  const char *start_good = jsonStr.c_str();
  const char *end_good = start_good + jsonStr.size();
  test_pointers(start_good, end_good);

  return 0;
}

你学废了吗?

相关推荐
AI小怪兽20 小时前
轻量、实时、高精度!MIE-YOLO:面向精准农业的多尺度杂草检测新框架 | MDPI AgriEngineering 2026
开发语言·人工智能·深度学习·yolo·无人机
码农小韩20 小时前
基于Linux的C++学习——循环
linux·c语言·开发语言·c++·算法
linweidong20 小时前
C++ 中避免悬挂引用的企业策略有哪些?
java·jvm·c++
CoderCodingNo20 小时前
【GESP】C++五级/四级练习(双指针/数学) luogu-P1147 连续自然数和
开发语言·c++·算法
IT=>小脑虎20 小时前
PHP零基础衔接进阶知识点【详解版】
开发语言·学习·php
颜酱20 小时前
前端算法必备:双指针从入门到很熟练(快慢指针+相向指针+滑动窗口)
前端·后端·算法
Wect20 小时前
LeetCode 274. H 指数:两种高效解法全解析
算法·typescript
Q741_14720 小时前
海致星图招聘 数据库内核研发实习生 一轮笔试 总结复盘(2) 作答语言:C/C++ 哈夫曼编码 LRU
c语言·数据库·c++·算法·笔试·哈夫曼编码·哈夫曼树
你怎么知道我是队长20 小时前
C语言---位域
c语言·开发语言
Hello.Reader20 小时前
PyFlink DataStream Operators 算子分类、函数写法、类型系统、链路优化(Chaining)与工程化踩坑
前端·python·算法