全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

C++如何实现一个简单的JSON解析器_从零开始编写C++递归下降JSON解析器

答案:本文介绍如何用C++从零实现一个简单的JSON解析器,核心是定义支持null、boolean、number、string、array和object的JsonValue类型,使用std::variant存储不同类型,并通过递归下降解析法为每种类型编写解析函数,最终组合成完整解析器。代码包含数据结构定义、解析逻辑和测试示例,适合理解JSON结构与编译原理基础。

想用C++从零实现一个简单的JSON解析器?关键在于理解JSON的结构和递归下降的基本思路。JSON支持几种基本类型:null、boolean、number、string、array 和 object,我们可以为每种类型写一个解析函数,通过递归组合起来。整个过程不需要复杂的工具,纯手工编写即可。

1. 定义JSON数据结构

首先,我们需要一个能表示任意JSON值的C++类型。使用 std::variant 是个好选择,它能安全地存储不同类型的数据。

#include 
#include 
#include 
#include 
#include 

// 前向声明解析器类 struct JsonValue;

using JsonObject = std::map; using JsonArray = std::vector; using JsonNumber = double; using JsonString = std::string;

// 核心类型:一个JSON值可以是 null, bool, number, string, array, object struct JsonValue { std::variant value;

// 构造函数
JsonValue() : value(nullptr) {}
JsonValue(std::nullptr_t) : value(nullptr) {}
JsonValue(bool b) : value(b) {}
JsonValue(double d) : value(d) {}
JsonValue(const std::string& s) : value(s) {}
JsonValue(JsonArray a) : value(std::move(a)) {}
JsonValue(JsonObject o) : value(std::move(o)) {}

// 类型判断辅助函数
bool is_null() const { return std::holds_alternative(value); }
bool is_bool() const { return std::holds_alternative(value); }
bool is_number() const { return std::holds_alternative(value); }
bool is_string() const { return std::holds_alternative(value); }
bool is_array() const { return std::holds_alternative(value); }
bool is_object() const { return std::holds_alternative(value); }

// 获取值(需确保类型正确)
bool as_bool() const { return std::get(value); }
double as_number() const { return std::get(value); }
const std::string& as_string() const { return std::get(value); }
const JsonArray& as_array() const { return std::get(value); }
const JsonObject& as_object() const { return std::get(value); }

};

2. 实现递归下降解析器

我们定义一个解析器类,维护当前解析位置和输入字符串。每个JSON类型对应一个解析函数。

class JsonParser {
private:
    const std::string& input;
    size_t pos;
// 跳过空白字符
void skip_whitespace() {
    while (pos < input.size() && (input[pos] == ' ' || input[pos] == '\t' || input[pos] == '\n' || input[pos] == '\r'))
        ++pos;
}

// 匹配并消耗指定字符
bool match(char c) {
    skip_whitespace();
    if (pos < input.size() && input[pos] == c) {
        ++pos;
        return true;
    }
    return false;
}

// 解析 "null"
JsonValue parse_null() {
    if (input.substr(pos, 4) == "null") {
        pos += 4;
        return JsonValue(nullptr);
    }
    throw std::runtime_error("Expected 'null'");
}

// 解析 "true" / "false"
JsonValue parse_boolean() {
    if (input.substr(pos, 4) == "true") {
        pos += 4;
        return JsonValue(true);
    }
    if (input.substr(pos, 5) == "false") {
        pos += 5;
        return JsonValue(false);
    }
    throw std::runtime_error("Expected 'true' or 'false'");
}

// 解析数字(简单版本,支持整数和小数)
JsonValue parse_number() {
    size_t start = pos;
    if (pos < input.size() && input[pos] == '-') ++pos;

    if (pos == input.size() || !isdigit(input[pos]))
        throw std::runtime_error("Invalid number");

    while (pos < input.size() && isdigit(input[pos])) ++pos;

    if (pos < input.size() && input[pos] == '.') {
        ++pos;
        if (pos == input.size() || !isdigit(input[pos]))
            throw std::runtime_error("Invalid number");
        while (pos < input.size() && isdigit(input[pos])) ++pos;
    }

    std::string numStr = input.substr(start, pos - start);
    try {
        return JsonValue(std::stod(numStr));
    } catch (...) {
        throw std::runtime_error("Invalid number format");
    }
}

// 解析带引号的字符串(未处理转义字符)
JsonValue parse_string() {
    if (!match('"')) throw std::runtime_error("Expected '\"'");

    size_t start = pos;
    while (pos < input.size() && input[pos] != '"') {
        ++pos;
    }

    if (pos == input.size()) throw std::runtime_error("Unterminated string");

    std::string str = input.substr(start, pos - start);
    ++pos; // 跳过结尾的 "
    return JsonValue(str);
}

// 解析数组
JsonValue parse_array() {
    if (!match('[')) throw std::runtime_error("Expected '['");

    JsonArray arr;
    skip_whitespace();

    if (match(']')) return JsonValue(arr); // 空数组

    while (true) {
        arr.push_back(parse_value());
        skip_whitespace();

        if (match(']')) break;
        if (!match(',')) throw std::runtime_error("Expected ',' or ']'");
    }

    return JsonValue(arr);
}

// 解析对象
JsonValue parse_object() {
    if (!match('{')) throw std::runtime_error("Expected '{'");

    JsonObject obj;
    skip_whitespace();

    if (match('}')) return JsonValue(obj); // 空对象

    while (true) {
        JsonValue keyVal = parse_string();
        std::string key = keyVal.as_string();

        skip_whitespace();
        if (!match(':')) throw std::runtime_error("Expected ':'");

        JsonValue value = parse_value();
        obj[key] = value;

        skip_whitespace();
        if (match('}')) break;
        if (!match(',')) throw std::runtime_error("Expected ',' or '}'");
    }

    return JsonValue(obj);
}

// 主解析入口
JsonValue parse_value() {
    skip_whitespace();
    if (pos >= input.size()) throw std::runtime_error("Unexpected end of input");

    char c = input[pos];
    if (c == 'n') return parse_null();
    if (c == 't' || c == 'f') return parse_boolean();
    if (c == '-' || isdigit(c)) return parse_number();
    if (c == '"') return parse_string();
    if (c == '[') return parse_array();
    if (c == '{') return parse_object();

    throw std::runtime_error("Unexpected character: " + std::string(1, c));
}

public: explicit JsonParser(const std::string& str) : input(str), pos(0) {}

JsonValue parse() {
    JsonValue result = parse_value();
    skip_whitespace();
    if (pos != input.size())
        throw std::runtime_error("Extra characters after JSON");
    return result;
}

};

3. 使用示例

写个简单的main函数测试一下:

#include 
#include 

void print_json(const JsonValue& val, int indent = 0) { std::string space(indent, ' '); if (val.is_null()) { std::cout << "null"; } else if (val.is_bool()) { std::cout << (val.as_bool() ? "true" : "false"); } else if (val.is_number()) { std::cout << val.as_number(); } else if (val.is_string()) { std::cout << "\"" << val.as_string() << "\""; } else if (val.is_array()) { std::cout << "[\n"; const auto& arr = val.as_array(); for (size_t i = 0; i < arr.size(); ++i) { std::cout << std::string(indent + 2, ' '); print_json(arr[i], indent + 2); if (i != arr.size() - 1) std::cout << ","; std::cout << "\n"; } std::cout << std::string(indent, ' ') << "]"; } else if (val.is_object()) { std::cout << "{\n"; const auto& obj = val.as_object(); auto it = obj.begin(); while (it != obj.end()) { std::cout << std::string(indent + 2, ' ') << "\"" << it->first << "\": "; print_json(it->second, indent + 2); ++it; if (it != obj.end()) std::cout << ","; std::cout << "\n"; } std::cout << std::string(indent, ' ') << "}"; } }

int main() { std::string json_str = R"({ "name": "Alice", "age": 30, "is_student": false, "grades": [85.5, 92.0, 78.5], "address": { "city": "Beijing", "zipcode": "100000" }, "spouse": null })";

try {
    JsonParser parser(json_str);
    JsonValue root = parser.parse();
    print_json(root);
    std::cout << std::endl;
} catch (const std::exception& e) {
    std::cerr << "Parse error: " << e.what() << std::endl;
    return 1;
}

return 0;

}

4. 注意事项与改进方向

这个解析器是教学级的,但已具备核心功能。你可以在此基础上扩展:

  • 支持字符串中的转义字符(如 \n, \", \\)
  • 支持科学计数法(如 1e5)
  • 添加位置信息便于报错定位
  • 实现序列化(将 JsonValue 转回字符串)
  • 优化性能,避免 substr 频繁拷贝

基本上就这些。递归下降的关键是“一个函数管一种语法结构”,逻辑清晰,容易调试。自己写一遍,对理解JSON和编译原理都很有帮助。


# js  # git  # json  # 工具  # ai  # c++  # ios  # stream  # String  # Boolean  # Array  # Object  # NULL  # const  # 字符串  # 递归  # 数据结构  # public  # number  # input  # 不同类型  # 跳过  # 是个  # 你可以  # 不需要  # 很有  # 一遍  # 我们可以 


相关文章: 如何在IIS7中新建站点?详细步骤解析  如何撰写建站申请书?关键要点有哪些?  如何快速查询网址的建站时间与历史轨迹?  建站之星代理如何获取技术支持?  杭州银行网站设计制作流程,杭州银行怎么开通认证方式?  如何快速完成中国万网建站详细流程?  网站图片在线制作软件,怎么在图片上做链接?  番禺网站制作公司哪家值得合作,番禺图书馆新馆开放了吗?  北京建设网站制作公司,北京古代建筑博物馆预约官网?  网站制作新手教程,新手建设一个网站需要注意些什么?  深圳网站制作平台,深圳市做网站好的公司有哪些?  微信小程序制作网站有哪些,微信小程序需要做网站吗?  深圳 网站制作,深圳招聘网站哪个比较好一点啊?  临沂网站制作公司有哪些,临沂第四中学官网?  建站之星3.0如何解决常见操作问题?  整蛊网站制作软件,手机不停的收到各种网站的验证码短信,是手机病毒还是人为恶搞?有这种手机病毒吗?  如何快速搭建虚拟主机网站?新手必看指南  图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?  贸易公司网站制作流程,出口贸易网站设计怎么做?  Python如何创建带属性的XML节点  建站主机功能解析:服务器选择与快速搭建指南  怎么用手机制作网站链接,dw怎么把手机适应页面变成网页?  网站制作多少钱一个,建一个论坛网站大约需要多少钱?  建站上传速度慢?如何优化加速网站加载效率?  模具网站制作流程,如何找模具客户?  如何通过虚拟主机快速搭建个人网站?  存储型VPS适合搭建中小型网站吗?  企业宣传片制作网站有哪些,传媒公司怎么找企业宣传片项目?  ,网站推广常用方法?  网站制作壁纸教程视频,电脑壁纸网站?  如何在云指建站中生成FTP站点?  深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?  如何通过宝塔面板实现本地网站访问?  宝盒自助建站智能生成技巧:SEO优化与关键词设置指南  浅谈Javascript中的Label语句  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?  手机网站制作与建设方案,手机网站如何建设?  高端网站建设与定制开发一站式解决方案 中企动力  如何在IIS管理器中快速创建并配置网站?  网页制作模板网站推荐,网页设计海报之类的素材哪里好?  如何在阿里云完成域名注册与建站?  广东专业制作网站有哪些,广东省能源集团有限公司官网?  浙江网站制作公司有哪些,浙江栢塑信息技术有限公司定制网站做的怎么样?  如何在建站之星网店版论坛获取技术支持?  如何用PHP工具快速搭建高效网站?  高防服务器租用指南:配置选择与快速部署攻略  Python文件管理规范_工程实践说明【指导】  如何快速搭建高效可靠的建站解决方案?  东莞市网站制作公司有哪些,东莞找工作用什么网站好? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。