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;
}
相关推荐
Vect__17 小时前
基于线程池从零实现TCP计算器网络服务
c++·网络协议·tcp/ip
naruto_lnq1 天前
分布式系统安全通信
开发语言·c++·算法
CSDN_RTKLIB1 天前
【四个场景测试】源文件编码UTF-8 BOM
c++
肉包_5111 天前
两个数据库互锁,用全局变量互锁会偶发软件卡死
开发语言·数据库·c++
Trouvaille ~1 天前
【Linux】UDP Socket编程实战(一):Echo Server从零到一
linux·运维·服务器·网络·c++·websocket·udp
HellowAmy1 天前
我的C++规范 - 线程池
开发语言·c++·代码规范
czy87874751 天前
const 在 C/C++ 中的全面用法(C/C++ 差异+核心场景+实战示例)
c语言·开发语言·c++
十五年专注C++开发1 天前
MinHook:Windows 平台下轻量级、高性能的钩子库
c++·windows·钩子技术·minhook
一只小小的芙厨1 天前
寒假集训笔记·树上背包
c++·笔记·算法·动态规划
以卿a1 天前
C++(继承)
开发语言·c++·算法