Qt 高效读写JSON文件,玩转QJsonDocument与QJsonObject

news2025/3/28 7:19:56

一、前言

JSON作为轻量级的数据交换格式,已成为开发者必备技能。Qt框架为JSON处理提供了完整的解决方案,通过QJsonDocumentQJsonObjectQJsonArray三大核心类,轻松实现数据的序列化与反序列化。

JSON vs INI

特性JSONINI
数据结构支持嵌套对象/数组扁平键值对
数据类型丰富(含null)仅字符串
适用场景复杂配置/网络传输简单配置

二、环境准备

2.1 项目配置

.pro文件中添加JSON模块:

QT += core

2.2 包含头文件

#include <QJsonDocument>
#include <QJsonObject>
#include <QJsonArray>
#include <QFile>

三、核心API详解

3.1 QJsonDocument类

方法说明示例
fromJson(jsonData, error)解析JSON数据QJsonParseError err; doc = QJsonDocument::fromJson(data, &err)
toJson(format)序列化为字符串QByteArray json = doc.toJson(QJsonDocument::Indented)
object()获取根对象QJsonObject root = doc.object()
array()获取根数组QJsonArray arr = doc.array()
isObject()是否对象类型if(doc.isObject())
isArray()是否数组类型if(doc.isArray())
setObject(obj)设置根对象doc.setObject(newObj)
setArray(arr)设置根数组doc.setArray(newArr)

3.2 QJsonObject类

方法说明示例
insert(key, value)插入键值对obj.insert("name", "Alice")
remove(key)删除指定键obj.remove("obsoleteKey")
contains(key)检查键是否存在if(obj.contains("timestamp"))
value(key)安全获取值QJsonValue val = obj.value("age")
operator[](key)直接访问值obj["score"] = 95.5
keys()获取所有键列表QStringList keys = obj.keys()
size()获取键值对数量qDebug() << "对象大小:" << obj.size()
isEmpty()判断是否为空if(obj.isEmpty()) return;

3.3 QJsonArray 类

方法说明示例
append(value)追加元素arr.append(42)
insert(index, value)插入元素arr.insert(0, "First")
removeAt(index)删除指定位置元素arr.removeAt(3)
replace(index, value)替换元素arr.replace(2, true)
at(index)获取指定位置值QJsonValue val = arr.at(0)
size()获取元素数量for(int i=0; i<arr.size(); ++i)
isEmpty()判断是否为空if(arr.isEmpty()) return;

3.4 QJsonValue类型处理

方法说明
isBool()布尔类型
isDouble()数值类型
isString()字符串类型
isArray()JSON数组
isObject()JSON对象
isNull()空值
isUndefined()未定义值

安全转换方法:

QJsonValue val = /* ... */;

// 带默认值的转换
int num = val.toInt(0); // 转换失败返回0
QString str = val.toString("default");

// 带错误检测的转换
bool ok;
double d = val.toDouble(&ok);
if(!ok) qWarning() << "转换double失败";

// 对象/数组转换
if(val.isObject()) {
    QJsonObject obj = val.toObject();
}
else if(val.isArray()) {
    QJsonArray arr = val.toArray();
}

// 特殊类型处理
QJsonValue nullVal = QJsonValue::Null;
QJsonValue undefinedVal; // 默认构造为Undefined

四、读写JSON

4.1 写入JSON文件,创建复杂JSON结构

void writeJsonFile()
{
    // 创建根对象
    QJsonObject rootObj;
    
    // 基础数据
    rootObj["appName"] = "QtConfigManager";
    rootObj["version"] = "1.2.0";
    rootObj["debugMode"] = false;

    // 嵌套对象
    QJsonObject dbConfig;
    dbConfig["host"] = "127.0.0.1";
    dbConfig["port"] = 3306;
    dbConfig["credentials"] = QJsonArray{"root", "123456"};
    rootObj["database"] = dbConfig;

    // 创建数组
    QJsonArray recentFiles;
    recentFiles.append("project1.pro");
    recentFiles.append("mainwindow.cpp");
    rootObj["recentFiles"] = recentFiles;

    // 生成JSON文档
    QJsonDocument doc(rootObj);
    
    // 写入文件
    QFile file("config.json");
    if(file.open(QIODevice::WriteOnly)){
        file.write(doc.toJson(QJsonDocument::Indented));
        file.close();
        qDebug() << "JSON文件写入成功!";
    } else {
        qWarning() << "文件打开失败:" << file.errorString();
    }
}

4.2 读取JSON文件

void readJsonFile()
{
    QFile file("config.json");
    if(!file.open(QIODevice::ReadOnly)) {
        qCritical() << "文件打开失败:" << file.errorString();
        return;
    }

    // 解析JSON
    QJsonDocument doc = QJsonDocument::fromJson(file.readAll());
    if(doc.isNull()){
        qWarning() << "JSON解析失败!";
        return;
    }

    // 获取根对象
    QJsonObject root = doc.object();
    
    // 读取基础配置
    QString appName = root["appName"].toString();
    bool debugMode = root["debugMode"].toBool();

    // 解析嵌套对象
    QJsonObject dbConfig = root["database"].toObject();
    QString host = dbConfig["host"].toString();
    QJsonArray credentials = dbConfig["credentials"].toArray();

    // 遍历数组
    QJsonArray files = root["recentFiles"].toArray();
    qDebug() << "最近文件列表:";
    for(const QJsonValue &val : files){
        qDebug() << "->" << val.toString();
    }

    // 安全取值示例
    int port = dbConfig.value("port").toInt(3306); // 默认值
}

五、实战技巧:处理复杂场景

5.1 动态键名处理

QJsonObject config;
QString dynamicKey = "custom_" + QString::number(QDateTime::currentSecsSinceEpoch());
config[dynamicKey] = "特殊配置项";

5.2 日期时间处理

// 写入
QJsonObject obj;
obj["timestamp"] = QDateTime::currentDateTime().toString(Qt::ISODate);

// 读取
QDateTime dt = QDateTime::fromString(obj["timestamp"].toString(), Qt::ISODate);

5.3 二进制数据编码

// Base64编码存储
QByteArray imageData = /*...*/;
obj["avatar"] = QString(imageData.toBase64());

// 解码读取
QByteArray restoredData = QByteArray::fromBase64(obj["avatar"].toString().toUtf8());

六、错误处理与调试

6.1 错误检测

QJsonParseError parseError;
QJsonDocument doc = QJsonDocument::fromJson(rawData, &parseError);

if(parseError.error != QJsonParseError::NoError){
    qDebug() << "JSON解析错误:" << parseError.errorString()
             << "at offset" << parseError.offset;
}

6.2 调试输出

// 格式化输出JSON
qDebug().noquote() << doc.toJson(QJsonDocument::Indented);

七、性能优化建议

  1. 大文件处理

    • 使用流式解析(QJsonDocument不适合GB级文件)

    • 考虑第三方库(如RapidJSON)处理超大JSON

  2. 内存管理

    // 及时释放不再使用的JSON对象
    {
        QJsonObject tempObj;
        // 操作临时对象
    } // 自动释放内存
  3. 缓存机制

    • 对频繁读取的配置进行内存缓存

    • 使用QCache实现LRU缓存

八、扩展应用:与QVariant互转

8.1 对象转换

// JSON转QVariantMap
QVariantMap vmap = doc.object().toVariantMap();

// QVariantMap转JSON
QJsonObject::fromVariantMap(vmap);

8.2 序列化对象

class UserSettings {
public:
    void saveToJson(QJsonObject &json) const {
        json["theme"] = m_theme;
        json["fontSize"] = m_fontSize;
    }
    
    void loadFromJson(const QJsonObject &json) {
        m_theme = json["theme"].toString();
        m_fontSize = json.value("fontSize").toInt(12);
    }
};

九、实践总结

  1. 文件操作规范

    • 使用QSaveFile实现原子写入

    • 设置文件权限:QFileDevice::ReadOwner | QFileDevice::WriteOwner

  2. 版本兼容性

    {
        "metadata": {
            "schemaVersion": "1.1",
            "createdAt": "2023-08-20"
        }
    }
  3. 安全建议

    • 校验JSON数据完整性

    • 限制最大文件尺寸

    • 敏感数据加密存储

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2321519.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

计算机网络——数据链路层的功能

目录 物理链路 逻辑链路 封装成帧&#xff08;组帧&#xff09; 帧定界 透明传输 SDU 差错控制 可靠传输 流量控制 介质访问控制 主机需要实现第一层到第五层的功能&#xff0c;而路由器这种节点只需要实现第一层到第三层的这些功能 假设左边用户需要给右边用户发送…

第60天:Web攻防-XSS跨站文件类型功能逻辑SVGPDFSWFPMessageLocalStorage

#知识点 1、Web攻防-XSS跨站-文件类型-html&pdf&swf&svg 2、Web攻防-XSS跨站-功能逻辑-postMessage&localStorage 术语&#xff1a;上传xss->其实就是将有恶意js代码的各类文件&#xff08;swf,pdf,svg,html.xml等&#xff09;上传->访问该文件->让浏…

C/C++都有哪些开源的Web框架?

CppCMS CppCMS是一个采用C语言开发的高性能Web框架&#xff0c;通过模版元编程方式实现了在编译期检查RESTful路由系统&#xff0c;支持传统的MVC模式和多种语言混合开发模式。 CppCMS最厉害的功能是WebSocket&#xff0c;10万连接在内存中长期保存占用的大小不超过600MB&…

RISC-V AIA学习2---IMSIC

我在学习文档这章时&#xff0c;对技术术语不太理解&#xff0c;所以用比较恰当的比喻来让自己更好的理解。 比较通俗的理解&#xff1a; 将 RISC-V 系统比作一个工厂&#xff1a; hart → 工厂的一条独立生产线IMSIC → 每条生产线配备的「订单接收员」MSI 中断 → 客户通过…

2024年MathorCup数学建模B题甲骨文智能识别中原始拓片单字自动分割与识别研究解题全过程文档加程序

2024年第十四届MathorCup高校数学建模挑战赛 B题 甲骨文智能识别中原始拓片单字自动分割与识别研究 原题再现&#xff1a; 甲骨文是我国目前已知的最早成熟的文字系统&#xff0c;它是一种刻在龟甲或兽骨上的古老文字。甲骨文具有极其重要的研究价值&#xff0c;不仅对中国文…

Python----计算机视觉处理(Opencv:霍夫变换)

一、霍夫变换 霍夫变换是图像处理中的一种技术&#xff0c;主要用于检测图像中的直线、圆或其他形状。其基本思想就是将图像空间中的点映射到参数空间中&#xff0c;通过在参数空间中寻找累计最大值来实现对特定形状的检测。 二、 霍夫直线变换 那么对于一个二值化后的图形来说…

多语言生成语言模型的少样本学习

摘要 大规模生成语言模型&#xff0c;如GPT-3&#xff0c;是极具竞争力的少样本学习模型。尽管这些模型能够共同表示多种语言&#xff0c;但其训练数据以英语为主&#xff0c;这可能限制了它们的跨语言泛化能力。在本研究中&#xff0c;我们在一个涵盖多种语言的语料库上训练了…

QT开发(4)--各种方式实现HelloWorld

目录 1. 编辑框实现 2. 按钮实现 前面已经写过通过标签实现的了&#xff0c;所以这里就不写了&#xff0c;通过这两个例子&#xff0c;其他的也是同理 1. 编辑框实现 编辑框分为单行编辑框&#xff08;QLineEdit&#xff09;双行编辑框&#xff08;QTextEdit&#xff09;&am…

Flutter 输入组件 Radio 详解

1. 引言 在 Flutter 中&#xff0c;Radio 是用于单选的按钮组件&#xff0c;适用于需要用户在多个选项中选择一个的场景&#xff0c;如表单、设置选项等。Radio 通过 value 和 groupValue 进行状态管理&#xff0c;并结合 onChanged 监听选中状态的变化。本文将介绍 Radio 的基…

3.23学习总结

完成了组合Ⅲ&#xff0c;和电话号码的字母组合两道算法题&#xff0c;都是和回溯有关的&#xff0c;很类似。 学习了static的关键字和继承有关知识

力扣刷题-热题100题-第23题(c++、python)

206. 反转链表 - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/reverse-linked-list/solutions/551596/fan-zhuan-lian-biao-by-leetcode-solution-d1k2/?envTypestudy-plan-v2&envIdtop-100-liked 常规法 记录前一个指针&#xff0c;当前指针&am…

vue3 项目的最新eslint9 + prettier 配置

注意&#xff1a;eslint目前升级到9版本了 在 ESLint v9 中&#xff0c;配置文件已经从 .eslintrc 迁移到了 eslint.config.js 配置的方式和之前的方式不太一样了&#xff01;&#xff01;&#xff01;&#xff01; 详见自己的语雀文档&#xff1a;5、新版eslint9prettier 配…

SAP GUI Script for C# SAP脚本开发快速指南与默认主题问题

SAP GUI Script for C# 快速指南 SAP 脚本的快速使用与设置. 解决使用SAP脚本执行后,默认打开的SAP是经典主题的问题 1. 解决默认主题问题 如果您使用的是SAP GUI 740&#xff0c;并遇到无法打开对话框的问题&#xff0c;请先将主题设置为经典主题&#xff08;Classic Theme…

FFmpeg + ‌Qt‌ 简单视频播放器代码

一个基于 ‌FFmpeg 4.x‌ 和 ‌Qt‌ 的简单视频播放器代码示例&#xff0c;实现视频解码和渲染到 Qt 窗口的功能。 1&#xff09;ffmpeg库界面&#xff0c;视频解码支持软解和硬解方式。 2&#xff09;QImage/QPixmap显示视频图片。 ‌1. Qt 项目配置&#xff08;.pro 文件&…

Unity跨平台构建快速回顾

知识点来源&#xff1a;人间自有韬哥在&#xff0c;豆包 目录 一、发布应用程序1. 修改发布必备设置1.1 打开设置面板1.2 修改公司名、游戏项目名、版本号和默认图标1.3 修改 Package Name 和 Minimum API Level 2. 发布应用程序2.1 配置 Build Settings2.2 选择发布选项2.3 构…

【嵌入式学习2】内存管理

## C语言编译过程 预处理&#xff1a;宏定义展开、头文件展开、条件编译&#xff0c;这里并不会检查语法&#xff0c;将#include #define这些头文件内容插入到源码中 gcc -E main.c -o main.i 编译&#xff1a;检查语法&#xff0c;将预处理后文件编译生成汇编文件&#xff…

TDengine又新增一可视化工具 Perspective

概述 Perspective 是一款开源且强大的数据可视化库&#xff0c;由 Prospective.co 开发&#xff0c;运用 WebAssembly 和 Web Workers 技术&#xff0c;在 Web 应用中实现交互式实时数据分析&#xff0c;能在浏览器端提供高性能可视化能力。借助它&#xff0c;开发者可构建实时…

【Linux文件IO】Linux中标准IO的API的描述和基本用法

Linux中标准IO的API的描述和基本用法 一、标准IO相关API1、文件的打开和关闭示例代码&#xff1a; 2、文件的读写示例代码&#xff1a;用标准IO&#xff08;fread、fwrite&#xff09;实现文件拷贝(任何文件均可拷贝) 3、文件偏移设置示例代码&#xff1a; 4、fgets fputs fget…

Ant Design Vue Select 选择器 全选 功能

Vue.js的组件库Ant Design Vue Select 选择器没有全选功能&#xff0c;如下图所示&#xff1a; 在项目中&#xff0c;我们自己实现了全选和清空功能&#xff0c;如下所示&#xff1a; 代码如下所示&#xff1a; <!--* 参数配置 - 风力发电 - 曲线图 * 猴王软件学院 - 大强 …

系统与网络安全------网络应用基础(1)

资料整理于网络资料、书本资料、AI&#xff0c;仅供个人学习参考。 TCP/IP协议及配置 概述 TCP/IP协议族 计算机之间进行通信时必须共同遵循的一种通信规定 最广泛使用的通信协议的集合 包括大量Internet应用中的标准协议 支持跨网络架构、跨操作系统平台的数据通信 主机…