Qt C++ :QJson使用详解

文章目录


Json的基本概念

Json 是一种数据格式,和语言无关,在什么语言中都可以使用 Json

Json中主要两种数据格式:Json数组Json对象

Json数组

  • Json数组使用 [] 表示,[] 里边是元素,元素和元素之间使用逗号间隔,最后一个元素后边没有逗号。
  • Json支持多种数据类型,包括:整形、 浮点型、 字符串、 布尔类型、 json数组、 json对象、 空值-null。
  • Json数组嵌套使用,Json数组中可以有Json数组。

Json对象

  • Json 对象使用 {} 来描述,每个 Json 对象中可以存储若干个元素,每一个元素对应一个键值(key:value 结构) 。
  • key 值必须是字符串,位于同一层级的键值不能重复(因为是通过键值取出对应的 value 值)。
  • value 值的类型是可选的,可根据实际需求指定,可用类型包括:整形、 浮点、 字符串、 布尔类型、 json数组、 json对象、 空值-null。

Qt 中 QJson 的核心类

Qt 提供了以下主要类用于处理 JSON:
QJsonDocument:表示整个 JSON 文档,是解析和生成 JSON 的入口。
QJsonObject:表示 JSON 对象(键值对集合,类似 std::map<QString, QJsonValue>)。
QJsonArray:表示 JSON 数组(有序的 QJsonValue 列表)。
QJsonValue:表示 JSON 中的一个值(可以是字符串、数字、布尔、null、对象或数组)。

所有字符串必须是 UTF-8 编码。

解析 JSON 字符串(反序列化)

cpp 复制代码
QString jsonString = R"({"name": "Alice", "age": 30, "hobbies": ["reading", "coding"]})";

QJsonDocument doc = QJsonDocument::fromJson(jsonString.toUtf8());
if (!doc.isNull() && doc.isObject()) {
    QJsonObject obj = doc.object();

    QString name = obj["name"].toString();
    int age = obj["age"].toInt();
    
    QJsonArray hobbies = obj["hobbies"].toArray();
    for (const QJsonValue &value : hobbies) {
        qDebug() << value.toString();
    }
}

注意:fromJson() 返回的文档若无效(如格式错误),isNull() 为 true。

生成 JSON 字符串(序列化)

cpp 复制代码
QJsonObject obj;
obj["name"] = "Bob";
obj["age"] = 25;

QJsonArray hobbies;
hobbies.append("gaming");
hobbies.append("music");
obj["hobbies"] = hobbies;

QJsonDocument doc(obj);
QString jsonString = QString::fromUtf8(doc.toJson(QJsonDocument::Compact));
// 或使用 QJsonDocument::Indented 输出带缩进的格式

QJsonDocument::toJson() 默认输出带换行和缩进,可用 QJsonDocument::Compact 减少体积。

嵌套结构处理示例

cpp 复制代码
// 构造 {"user": {"id": 1, "profile": {"email": "a@example.com"}}}
QJsonObject profile;
profile["email"] = "a@example.com";

QJsonObject user;
user["id"] = 1;
user["profile"] = profile;

QJsonObject root;
root["user"] = user;

解析时同样逐层访问:

cpp 复制代码
QJsonValue userVal = root["user"];
if (userVal.isObject()) {
    QJsonObject userObj = userVal.toObject();
    int id = userObj["id"].toInt();
    QJsonObject profileObj = userObj["profile"].toObject();
    QString email = profileObj["email"].toString();
}

判断值类型与安全访问

QJsonValue 提供类型判断方法:

cpp 复制代码
QJsonValue val = obj["key"];
if (val.isString()) { ... }
if (val.isDouble()) { ... } // 注意:整数在 JSON 中也是 double 类型
if (val.isArray()) { ... }
if (val.isObject()) { ... }
if (val.isNull()) { ... }   // 表示 null 或 key 不存在(需注意区别)

注意:如果访问一个不存在的 key,QJsonValue 会返回 QJsonValue::Undefined,其 isNull() 返回 false,但 isUndefined() 为 true。建议先用 contains() 检查 key 是否存在:

cpp 复制代码
if (obj.contains("name")) {
    QString name = obj["name"].toString();
}

错误处理建议

使用 QJsonParseError 获取解析错误详情:

cpp 复制代码
QJsonParseError error;
QJsonDocument doc = QJsonDocument::fromJson(jsonData, &error);
if (doc.isNull()) {
    qDebug() << "JSON 解析错误:" << error.errorString()
             << "位置:" << error.offset;
}
相关推荐
澈2074 小时前
深入浅出C++滑动窗口算法:原理、实现与实战应用详解
数据结构·c++·算法
A.A呐4 小时前
【C++第二十九章】IO流
开发语言·c++
ambition202425 小时前
从暴力搜索到理论最优:一道任务调度问题的完整算法演进历程
c语言·数据结构·c++·算法·贪心算法·深度优先
kebeiovo5 小时前
atomic原子操作实现无锁队列
服务器·c++
Yungoal5 小时前
常见 时间复杂度计算
c++·算法
6Hzlia5 小时前
【Hot 100 刷题计划】 LeetCode 48. 旋转图像 | C++ 矩阵变换题解
c++·leetcode·矩阵
Ricky_Theseus6 小时前
C++右值引用
java·开发语言·c++
吴梓穆7 小时前
UE5 c++ 常用方法
java·c++·ue5
云栖梦泽7 小时前
Linux内核与驱动:9.Linux 驱动 API 封装
linux·c++
Morwit7 小时前
【力扣hot100】 1. 两数之和
数据结构·c++·算法·leetcode·职场和发展