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;
}
相关推荐
u0109272711 天前
C++中的策略模式变体
开发语言·c++·算法
Aevget1 天前
MFC扩展库BCGControlBar Pro v37.2新版亮点:控件功能进一步升级
c++·mfc·界面控件
四维碎片1 天前
QSettings + INI 笔记
笔记·qt·算法
Tansmjs1 天前
C++与GPU计算(CUDA)
开发语言·c++·算法
挖矿大亨1 天前
c++中的函数模版
java·c++·算法
阿基米东1 天前
基于 C++ 的机器人软件框架(具身智能)开源通信库选型分析
c++·机器人·开源
偷星星的贼111 天前
C++中的对象池模式
开发语言·c++·算法
CN-Dust1 天前
【C++】洛谷P3073 [USACO13FEB] Tractor S
开发语言·c++
2401_829004021 天前
C++中的适配器模式变体
开发语言·c++·算法
平生不喜凡桃李1 天前
二叉树遍历非递归写法: 栈
c++··二叉树遍历·非递归