简单实现QT对象的[json]序列化与反序列化
简单实现QT对象的[json]序列化与反序列化
- 简介
- 应用场景
- qt元对象系统
- 思路
- 实现
- 使用方式
- 题外话
简介
众所周知json作为一种轻量级的数据交换格式,在开发中被广泛应用。因此如何方便的将对象数据转为json格式和从json格式中加载数据到对象中就变得尤为重要。
在python类动态语言中,我们可以很方便的使用json.dumps()和json.load()完成json数据的生成和加载。但是在QT中就不能非常方便的转换。
因此本文将使用QT中的元对象系统实现简单的json数据转换和加载
应用场景
假如现在需要开发一个系统,采用c/s架构,使用json格式作为前后端的数据交互协议,那么针对每个接口,所需要交互的数据都需要转换为json或者从json中获取。
举个栗子:
现有个需要,需要根据用户id获取用户的详细信息,由客户端发起请求,服务端返回数据
//客户端
//1.定义数据结构体
struct requestUserInfo
{int id = -1;
};
struct responseUserInfo
{QString name;int age;//....
};
//2.定义数据
requestUserInfo _data;
_data.id = 1;
//3.转换为json数据,转换过程就不展示了
QByteArray _requestData = "xxx";
auto _result = network.sendData(_requestData);
//4.将结果转换为responseUserInfo,转换过程就不详细描述了
responseUserInfo _responseData;
//5.显示详细信息
//.....
在这个过程中,可以很清楚的看到,第3步和第4步的转换和解析过程将会耗费大量的事件书写很多冗余的代码,如果能够很方便的将结构体(对象)转换为json 和从json转换为结构体。代码的书写将会变得简洁而高效。
qt元对象系统
在此之前我们需要先了解一下qt的元对象系统是什么,ai回答如下:
- 信号和槽机制:Qt的元对象系统支持信号和槽之间的通信,这是Qt中对象间通信的主要方式。信号是由对象发出的通知,告知发生了某个事件,而槽是响应这些信号的函数。这种机制替代了传统的回调函数,使得对象间的协作更加简单和直观
- 运行时类型信息:元对象系统提供了运行时类型信息(RTTI),允许程序在运行时查询对象的类型。这包括获取类的名称、父类的名称、类中信号和槽的数量和名称等
- 动态属性系统:Qt的元对象系统支持动态属性,这意味着可以在运行时添加、修改或删除对象的属性。这些属性可以用于Qt Designer工具中,也可以在QML中使用
- 继承和多态:Qt的元对象系统支持面向对象编程中的继承和多态特性。通过使用Q_OBJECT宏和元对象编译器(MOC),Qt能够为继承自QObject的类生成额外的代码,以支持信号和槽、属性等元对象特性
- 元对象编译器(MOC):为了使用元对象系统,需要在类定义中包含Q_OBJECT宏,然后使用MOC编译器生成额外的代码来支持元对象特性。MOC是Qt构建过程中自动调用的,它解析C++头文件,并为包含Q_OBJECT宏的类生成运行时所需的代码
- 对象树和所有权:Qt使用对象树来组织和管理所有的QObject及其子类的对象。当一个对象被创建并指定另一个对象为其父对象时,它会被添加到父对象的孩子列表中。当父对象被销毁时,子对象也会自动被销毁,这有助于简化内存管理并减少内存泄漏的风险
- 自定义属性:Qt提供了基于元对象系统的自定义属性机制,允许开发者定义自己的属性,并在Qt Designer和QML中使用这些属性。这些属性可以通过Q_PROPERTY宏来声明,并可以设置为动态属性
总之,继承至QObject的子类,在运行时可以通过qt的元对象系统获取对象的名称和属性信息,并且可以自定义信息,其他的特性不在本文讨论的范围中,如果大家感兴趣,后续可以专门写篇文章介绍。
思路
上面提到Qt的元对象系统可以在运行时获取对象的属性,并且支持自定义属性,所以我们就需要借助元对象的这两个属性来完成对象的序列化和反序列化。
在利用qt的元对象特性,利用QMetaObject类获取属性名称和属性值,在转换为json格式,同理从json格式加载时,也可以利用QMetaObject类向属性中写入数据。
实现
通过上述的原理,我实现了简单的转换和加载函数,都包含在一个头文件中,使用时也只需要包含头文件即可,还是很方便。
jsonHelper.h
#include <QObject>
#include <QJsonObject>
#include <QMetaObject>
#include <QMetaProperty>
#include <QMetaMethod>
#include <QJsonValue>#define READ_WRITE_VALUE(_type,val) \
_type get_##val() const {return val;} \
void set_##val(const _type& _t){val = _t;}#define READ_WRITE_VALUE_TEMPLATE(val) \
auto get_##val() const -> decltype(val) { return val; } \
void set_##val(const decltype(val)& _t) { val = _t; }#define READ_FUNC_NAME(_name) get_##_name
#define WRITE_FUNC_NAME(_name) set_##_name#define READ_OBJECT(_name) Q_INVOKABLE QJsonObject read_##_name(){return objDump(&_name);}#define WRITE_OBJECT(_name) Q_INVOKABLE void set_##_name(const QJsonObject& _jsonObj){ objLoad(_jsonObj,&_name);}inline QJsonObject objDump(QObject* _obj)
{QJsonObject _jsonObj;auto _meatObj = _obj->metaObject();for(int i = _meatObj->propertyOffset(); i < _meatObj->propertyCount(); i++){auto _property = _meatObj->property(i);auto _type = _property.type();if(_type < QVariant::UserType){_jsonObj[_property.name()] = _property.read(_obj).toJsonValue();}}for(int i = _meatObj->methodOffset(); i < _meatObj->methodCount(); i++){auto _meth = _meatObj->method(i);if(_meth.returnType() != QMetaType::QJsonObject){continue;}QJsonObject _tempObj;_meth.invoke(_obj,Qt::AutoConnection,Q_RETURN_ARG(QJsonObject,_tempObj));QString _key = _meth.name();_key = _key.remove("read_");_jsonObj[_key] = _tempObj;}return _jsonObj;
}inline void objLoad(const QJsonObject& jsonObj,QObject* _obj)
{auto _meatObj = _obj->metaObject();for(int i = _meatObj->propertyOffset(); i < _meatObj->propertyCount(); i++){auto _property = _meatObj->property(i);QJsonValue _val = jsonObj[_property.name()];_property.write(_obj,_val.toVariant());}for(int i = _meatObj->methodOffset(); i < _meatObj->methodCount(); i++){auto _meth = _meatObj->method(i);if(_meth.returnType() != QMetaType::Void){continue;}QString _key = _meth.name();_key = _key.remove("set_");QJsonObject tempObj = jsonObj[_key].toObject();_meth.invoke(_obj,Q_ARG(QJsonObject,tempObj));}
}
使用方式
- 定义数据结构对象
#include "JsonHelper.h"class testStruct : public QObject
{Q_OBJECTQ_PROPERTY(int a READ READ_FUNC_NAME(a) WRITE WRITE_FUNC_NAME(a) CONSTANT)
public:testStruct &operator=(const testStruct& other){if(this != &other){this->a = other.a;}return *this;}int a;protected:READ_WRITE_VALUE(int,a)
};class myStruct : public QObject
{Q_OBJECTQ_PROPERTY(int id READ READ_FUNC_NAME(id) WRITE WRITE_FUNC_NAME(id) CONSTANT)Q_PROPERTY(QString name READ READ_FUNC_NAME(name) WRITE WRITE_FUNC_NAME(name) CONSTANT)Q_PROPERTY(QStringList nameList READ READ_FUNC_NAME(nameList) WRITE WRITE_FUNC_NAME(nameList) CONSTANT)
public:int id;QString name;QStringList nameList;testStruct _s1;protected:READ_WRITE_VALUE(int,id)READ_WRITE_VALUE(QString,name)READ_WRITE_VALUE(QStringList,nameList)READ_OBJECT(_s1)WRITE_OBJECT(_s1)};
- 类型转换与加载
void int main()
{testStruct _ss;_ss.a = 12;myStruct _t;_t.id = 1;_t.name = "小明";_t.nameList = QStringList({"zhangsan","lisi"});_t._s1 = _ss;auto _obj = objDump(&_t);qDebug() << _obj;myStruct _t1;objLoad(_obj,&_t1);qDebug() << _t1.id << _t1.name << _t1.nameList << _t1._s1.a;return 0;
}
题外话
可以发现定义数据结构时还是比较繁琐的,如果有更方便的方式来定义数据结构,在实际场景中可用性将会变的更好。
因此,我实现了一个简单的代码自动生成工具,可以将struct结构体转换为满足上述条件的数据对象。如下所示。

这样就可以不用费劲的编写数据对象结构了。
工具和代码都已经上传GitHub了,感兴趣可以下载使用
GitHub链接
也提供了打包好的生成工具,可直接下载使用:
生成工具链接
相关文章:
简单实现QT对象的[json]序列化与反序列化
简单实现QT对象的[json]序列化与反序列化 简介应用场景qt元对象系统思路实现使用方式题外话 简介 众所周知json作为一种轻量级的数据交换格式,在开发中被广泛应用。因此如何方便的将对象数据转为json格式和从json格式中加载数据到对象中就变得尤为重要。 在python类…...
Unity肢体控制(关节控制)
前面的基础搭建网上自己搜,我这个任务模型网上也有,可以去官网看看更多模型,这里只讲述有模型如何驱动肢体的操作方式 第一步:创建脚本 第二步:创建Rig Builder 建空容器 加部件(Rig),加了之后…...
Node.js | Yarn下载安装与环境配置
一、安装Node.js Yarn 是 Node.js 下的包管理工具,因此想要使用 Yarn 就必须先下载 Node.js。 推荐参考:Node.js | npm下载安装及环境配置教程 二、Yarn安装 打开cmd,输入以下命令: npm install -g yarn检查是否安装成功&…...
WPF如何全局应用黑白主题效果
灰白色很多时候用于纪念,哀悼等。那么使用 WPF如何来做到这种效果呢?要实现的这种效果,我们会发现,它其实不仅仅是要针对图片,而是要针对整个窗口来实现灰白色。 如果只是针对图片的话,我可以可以对图片进…...
[Qt] Qt删除文本文件中的某一行
需求 我们经常读一个文件或者直接往一个空白文件中写文本,那么该如何使用Qt在一个文本文件中删除某一行 代码 #include <QCoreApplication> #include <QIODevice> #include <QFile> #include <QTextStream> #include <QString> #i…...
【HarmonyOS学习日志(9)】一次开发,多端部署之界面级一多开发
关于一次开发,多端部署 一次开发多端部署就是指一套代码工程,一次开发上架,多端按需部署(一多),用于支撑开发者快速高效地开发多终端设备上的应用,以节省开发成本。 HarmonyOS系统面向多终端&…...
基于Java+SSM+JSP+MYSQL实现的宠物领养收养管理系统功能设计与实现六
一、前言介绍: 免费学习:猿来入此 1.1 项目摘要 随着人们生活水平的提高,宠物已经成为越来越多家庭的重要成员。然而,宠物的数量增长也带来了一系列问题,如流浪宠物数量的增加、宠物健康管理的缺失以及宠物领养收养…...
Java项目实战II基于微信小程序的课堂助手(开发文档+数据库+源码)
目录 一、前言 二、技术介绍 三、系统实现 四、文档参考 五、核心代码 六、源码获取 全栈码农以及毕业设计实战开发,CSDN平台Java领域新星创作者,专注于大学生项目实战开发、讲解和毕业答疑辅导。获取源码联系方式请查看文末 一、前言 在数字化教…...
解析 Android WebChromeClient:提升 WebView 用户体验的关键组件
文章目录 一、总览二、详细说明三、一些实际和有趣的应用四、最佳实践五、与其他组件的比较六、安全性考虑:防止 XSS 攻击与数据泄露6.1 介绍6.2 代码案例6.2.1 输入过滤6.2.2 Content Security Policy (CSP) 案例 六、总结 在 Android 开发中,WebChrome…...
【LeetCode热题100】字符串
本篇博客记录了关于字符串相关的几道题目,包括最长公共前缀、最长回文子串、二进制求和、字符串相乘。 //解法1 class Solution { public:string longestCommonPrefix(vector<string>& strs) {string ret strs[0];for(int i 1 ; i < strs.size() ; i…...
OceanBase 闪回查询
前言 在OB中,drop表可以通过 回收站 或者 以往的备份恢复来还原单表。当delete数据时,由于delete操作的对象不会进入回收站,此时需要通过闪回查询功能查看delete的数据,以便后续恢复 本次实验版本为 OceanBase 4.2.1.8࿰…...
C++析构函数详解
C析构函数详解:对象销毁与资源清理 在 C 中,析构函数是与构造函数相对应的特殊成员函数,它在对象生命周期结束时被自动调用,用于执行对象销毁之前的清理操作。析构函数主要用于释放对象占用的资源,如动态分配的内存、打…...
【网络安全 | 漏洞挖掘】未授权获取AI聊天内容
未经许可,不得转载。 文章目录 两天前,我收到了一项私人项目的邀请,内容看起来像是一个聊天机器人,类似于 Gemini 或 ChatGPT。于是我开始测试该项目的一些业务逻辑漏洞和 IDOR(不当访问控制)漏洞。尽管这个产品拥有一个强大的安全团队,网站上也部署了 WAF(Web 应用防火…...
时间序列分析——移动平均法、指数平滑法、逐步回归法、趋势外推法等(基于Python实现)
第 11章——时间序列分析和预测 【例11-1】 绘制时间序列折线图—观察成分 【代码框11-1】——绘制时间序列折线图 # 图11-2的绘制代码 import pandas as pd import matplotlib.pyplot as plt plt.rcParams[font.sans-serif]=[SimHei...
opencv(c++)----图像的读取以及显示
opencv(c)----图像的读取以及显示 imread: 作用:读取图像文件并将其加载到 Mat 对象中。参数: 第一个参数是文件路径,可以是相对路径或绝对路径。第二个参数是读取标志,比如 IMREAD_COLOR 表示以彩色模式读取图像。 返回值&#x…...
PyTorch——从入门到精通:PyTorch基础知识(张量)【PyTorch系统学习】
什么是张量(Tensor) 张量在数学中是一个代数对象,描述了与矢量空间相关的代数对象集之间的多重线性映射。张量是向量和矩阵概念的推广,可以理解为多维数组。作为数学中的一个基本概念,张量有着多种类型,…...
(笔记)ubuntu20安装jdk7,多版本管理
前往 Oracle JDK 7 下载页面(需要 Oracle 账户),下载 JDK 7 的压缩包文件(.tar.gz)。 下载完成后,将文件解压到 /opt 目录: sudo tar -xzf jdk-7u<version>-linux-x64.tar.gz -C /opt 重…...
Python系列教程
文章目录 1. Python基础2. Python基础库3. Python数据分析 1. Python基础 语句数据类型表达式输入、输出与文件读写函数模块与包类与面向对象作用域与命名空间常用技巧与操作 2. Python基础库 Typing库 3. Python数据分析...
如何恢復電腦IP地址的手動設置?
手動設置IP地址後,可能會遇到一些網路連接問題,或者需要恢復到之前的自動獲取狀態。這篇文章將詳細介紹如何恢復電腦的IP地址設置。 為什麼需要恢復IP地址設置? 網路連接問題:手動設置IP地址後,可能會導致與路由器或…...
Linux 下敏感文件路径总结
Linux 下敏感文件路径总结 在服务器运维和安全测试过程中,掌握各类服务的关键配置文件路径、日志文件位置以及重要目录的存放位置至关重要。本文整理了 Linux 系统下常见服务(如 Apache、Nginx、MySQL 等)的路径结构,以及一些敏感…...
挑战杯推荐项目
“人工智能”创意赛 - 智能艺术创作助手:借助大模型技术,开发能根据用户输入的主题、风格等要求,生成绘画、音乐、文学作品等多种形式艺术创作灵感或初稿的应用,帮助艺术家和创意爱好者激发创意、提高创作效率。 - 个性化梦境…...
XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...
《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》
引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...
Cesium1.95中高性能加载1500个点
一、基本方式: 图标使用.png比.svg性能要好 <template><div id"cesiumContainer"></div><div class"toolbar"><button id"resetButton">重新生成点</button><span id"countDisplay&qu…...
Python如何给视频添加音频和字幕
在Python中,给视频添加音频和字幕可以使用电影文件处理库MoviePy和字幕处理库Subtitles。下面将详细介绍如何使用这些库来实现视频的音频和字幕添加,包括必要的代码示例和详细解释。 环境准备 在开始之前,需要安装以下Python库:…...
【C语言练习】080. 使用C语言实现简单的数据库操作
080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...
NFT模式:数字资产确权与链游经济系统构建
NFT模式:数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新:构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议:基于LayerZero协议实现以太坊、Solana等公链资产互通,通过零知…...
QT: `long long` 类型转换为 `QString` 2025.6.5
在 Qt 中,将 long long 类型转换为 QString 可以通过以下两种常用方法实现: 方法 1:使用 QString::number() 直接调用 QString 的静态方法 number(),将数值转换为字符串: long long value 1234567890123456789LL; …...
Java线上CPU飙高问题排查全指南
一、引言 在Java应用的线上运行环境中,CPU飙高是一个常见且棘手的性能问题。当系统出现CPU飙高时,通常会导致应用响应缓慢,甚至服务不可用,严重影响用户体验和业务运行。因此,掌握一套科学有效的CPU飙高问题排查方法&…...
LeetCode - 199. 二叉树的右视图
题目 199. 二叉树的右视图 - 力扣(LeetCode) 思路 右视图是指从树的右侧看,对于每一层,只能看到该层最右边的节点。实现思路是: 使用深度优先搜索(DFS)按照"根-右-左"的顺序遍历树记录每个节点的深度对于…...
