C++中使用Qt实现JSON序列化与反序列化

// File: JsonSerializer
// Author: linxmouse@gmail.com
// Creation: 2024/09/29
#ifndef JSON_SERIALIZER_H
#define JSON_SERIALIZER_H

#include <QJsonDocument>
#include <QJsonObject>
#include <QJsonArray>
#include <QJsonValue>
#include <type_traits>

/* META OBJECT SYSTEM */
#include <QVariant>
#include <QMetaProperty>
#include <QMetaObject>
#include <QMetaType>

/* CONTAINER TYPE */
#include <QVector>
#include <QList>
#include <QMap>
#include <QHash>
#include <vector>
#include <map>

// Forward declarations
template<typename T, typename Enable = void>
struct Serializer;

// Primary template for QJsonValue conversion
template<typename T>
struct ToJsonValue {
	static QJsonValue convert(const T& value) {
		return QJsonValue::fromVariant(QVariant::fromValue(value));
	}
};

// Specialization for QJsonValue
template<>
struct ToJsonValue<QJsonValue> {
	static QJsonValue convert(const QJsonValue& value) {
		return value;
	}
};

// Serializer for primitive types
template<typename T>
struct Serializer<T, typename std::enable_if<std::is_arithmetic<T>::value || std::is_same<T, QString>::value>::type> {
	static QJsonValue toJson(const T& value) {
		return ToJsonValue<T>::convert(value);
	}

	static T fromJson(const QJsonValue& json) {
		return json.toVariant().value<T>();
	}
};

// Serializer for QList and QVector
template<template<typename> class Container, typename T>
struct Serializer<Container<T>, typename std::enable_if<std::is_same<Container<T>, QList<T>>::value || std::is_same<Container<T>, QVector<T>>::value>::type> {
	static QJsonValue toJson(const Container<T>& container) {
		QJsonArray array;
		for (const auto& item : container) {
			array.append(Serializer<T>::toJson(item));
		}
		return array;
	}

	static Container<T> fromJson(const QJsonValue& json) {
		Container<T> result;
		if (json.isArray()) {
			QJsonArray array = json.toArray();
			for (const auto& item : array) {
				result.append(Serializer<T>::fromJson(item));
			}
		}
		return result;
	}
};

// Serializer for std::vector
template<typename T>
struct Serializer<std::vector<T>> {
	static QJsonValue toJson(const std::vector<T>& container) {
		QJsonArray array;
		for (const auto& item : container) {
			array.append(Serializer<T>::toJson(item));
		}
		return array;
	}

	static std::vector<T> fromJson(const QJsonValue& json) {
		std::vector<T> result;
		if (json.isArray()) {
			QJsonArray array = json.toArray();
			for (const auto& item : array) {
				result.push_back(Serializer<T>::fromJson(item));
			}
		}
		return result;
	}
};

// Serializer for QMap and QHash
template<template<typename, typename> class Map, typename K, typename V>
struct Serializer<Map<K, V>, typename std::enable_if<std::is_same<Map<K, V>, QMap<K, V>>::value || std::is_same<Map<K, V>, QHash<K, V>>::value>::type> {
	static QJsonValue toJson(const Map<K, V>& map) {
		QJsonObject obj;
		for (auto it = map.begin(); it != map.end(); ++it) {
			obj.insert(ToJsonValue<K>::convert(it.key()).toString(), Serializer<V>::toJson(it.value()));
		}
		return obj;
	}

	static Map<K, V> fromJson(const QJsonValue& json) {
		Map<K, V> result;
		if (json.isObject()) {
			QJsonObject obj = json.toObject();
			for (auto it = obj.begin(); it != obj.end(); ++it) {
				result.insert(ToJsonValue<K>::convert(it.key()).toVariant().value<K>(),
					Serializer<V>::fromJson(it.value()));
			}
		}
		return result;
	}
};

// Serializer for std::map
template<typename K, typename V>
struct Serializer<std::map<K, V>> {
	static QJsonValue toJson(const std::map<K, V>& map) {
		QJsonObject jsonObject;
		for (const auto& pair : map) {
			QString key = ToJsonValue<K>::convert(pair.first).toString();
			QJsonValue value = Serializer<V>::toJson(pair.second);
			jsonObject.insert(key, value);
		}
		return jsonObject;
	}

	static std::map<K, V> fromJson(const QJsonValue& json) {
		std::map<K, V> result;
		if (json.isObject()) {
			QJsonObject jsonObject = json.toObject();
			for (auto it = jsonObject.begin(); it != jsonObject.end(); ++it) {
				K key = ToJsonValue<K>::convert(it.key()).toVariant().value<K>();
				V value = Serializer<V>::fromJson(it.value());
				result.insert({ key, value });
			}
		}
		return result;
	}
};

#define JSON_SERIALIZABLE \
    virtual const QMetaObject* metaObject() const { \
    \
        return &this->staticMetaObject; \
    }

// Base class for serializable objects
class JsonSerializable {
	Q_GADGET
	JSON_SERIALIZABLE
public:
	virtual ~JsonSerializable() = default;
	// Convert QJsonValue in QJsonDocument as QByteArray.
	static QByteArray toByteArray(const QJsonValue& value) {
		return QJsonDocument(value.toObject()).toJson();
	}
	// Serialize all accessed JSON propertyes for this object.
	QJsonObject toJson() const {
		QJsonObject json;
		int propCount = metaObject()->propertyCount();
		for (int i = 0; i < propCount; i++)
		{
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
			if (QString(metaObject()->property(i).typeName()) != QMetaType::typeName(qMetaTypeId<QJsonValue>())) {
				continue;
			}
#else
			if (metaObject()->property(i).metaType().id() != QMetaType::QJsonValue) {
				continue;
			}
#endif

			auto key = metaObject()->property(i).name();
			auto value = metaObject()->property(i).readOnGadget(this).toJsonValue();
			json.insert(key, value);
		}
		return json;
	}

	// Returns QByteArray representation this object using json-serialization.
	QByteArray toRawJson() const {
		return toByteArray(toJson());
	}

	// Deserialize all accessed JSON propertyes for this object.
	void fromJson(const QJsonValue& val) {
		if (val.isObject())
		{
			QJsonObject json = val.toObject();
			QStringList keys = json.keys();
			int propCount = metaObject()->propertyCount();
			for (int i = 0; i < propCount; i++)
			{
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
				if (QString(metaObject()->property(i).typeName()) != QMetaType::typeName(qMetaTypeId<QJsonValue>())) {
					continue;
				}
#else
				if (metaObject()->property(i).metaType().id() != QMetaType::QJsonValue) {
					continue;
				}
#endif

				QString propertyName = metaObject()->property(i).name();
				for (const auto& key : json.keys())
				{
					if (key.compare(propertyName, Qt::CaseInsensitive) == 0)
					{
						metaObject()->property(i).writeOnGadget(this, json.value(key));
						break;
					}
				}
			}
		}
	}
	// Deserialize all accessed JSON propertyes for this object.
	void fromJson(const QByteArray& data) {
		fromJson(QJsonDocument::fromJson(data).object());
	}
};

// Serializer for custom types inheriting from JsonSerializable
template<typename T>
struct Serializer<T, typename std::enable_if<std::is_base_of<JsonSerializable, T>::value>::type> {
	static QJsonValue toJson(const T& value) {
		return value.toJson();
	}

	static T fromJson(const QJsonValue& json) {
		T result;
		result.fromJson(json.toObject());
		return result;
	}
};

// Macros for easy property declaration
#define JSON_PROPERTY(type, name) \
    Q_PROPERTY(QJsonValue name READ get_##json##_##name WRITE set_##json##_##name) \
    private: \
        type m_##name; \
        QJsonValue get_##json##_##name() const { return Serializer<type>::toJson(m_##name); } \
        void set_##json##_##name(const QJsonValue& value) { m_##name = Serializer<type>::fromJson(value); } \
    public: \
        type name() const { return m_##name; } \
        void set_##name(const type& value) { m_##name = value; }

#endif // JSON_SERIALIZER_H

在现代软件开发中,JSON(JavaScript Object Notation)已成为一种广泛使用的数据交换格式。本文将介绍如何在C++中使用Qt框架实现一个灵活且易用的JSON序列化和反序列化系统。

核心思想

这个序列化系统的核心思想是:

  1. 使用模板和特化来处理不同类型的序列化和反序列化。
  2. 利用Qt的元对象系统来自动化序列化过程。
  3. 提供一个基类JsonSerializable和宏来简化使用。

关键组件

1. Serializer模板

template<typename T, typename Enable = void>
struct Serializer;

这是整个系统的核心。它为不同的类型提供了toJsonfromJson方法。通过特化这个模板,我们可以为不同的类型定义自定义的序列化行为。

2. ToJsonValue

template<typename T>
struct ToJsonValue {
    static QJsonValue convert(const T& value) {
        return QJsonValue::fromVariant(QVariant::fromValue(value));
    }
};

这个辅助结构体用于将各种类型转换为QJsonValue

3. JsonSerializable基类

class JsonSerializable {
    Q_GADGET
    JSON_SERIALIZABLE
public:
    QJsonObject toJson() const;
    void fromJson(const QJsonValue& val);
    // ...其他方法...
};

这个基类提供了基本的序列化和反序列化功能。它利用Qt的元对象系统来自动化处理对象的属性。

4. JSON_PROPERTY宏

#define JSON_PROPERTY(type, name) \
    // ... 宏定义 ...

这个宏简化了在派生类中声明可序列化属性的过程。

支持的类型

这个系统支持多种类型:

  • 基本类型 (int, float, QString等)
  • Qt容器 (QList, QVector, QMap, QHash)
  • 标准库容器 (std::vector, std::map)
  • 自定义类型 (只要它们从JsonSerializable派生)

优点

  1. 类型安全: 使用模板和类型特化确保类型安全。
  2. 灵活性: 容易扩展以支持新的类型。
  3. 易用性: 使用宏和基类简化了使用过程。
  4. 自动化: 利用Qt的元对象系统自动处理属性。

使用方法:序列化和反序列化

#ifndef TEST_PERSON_H
#define TEST_PERSON_H

#include "JsonSerializer.h"

class TestPerson final: public JsonSerializable
{
	Q_GADGET
	JSON_SERIALIZABLE
public:
	JSON_PROPERTY(QString, name)
	JSON_PROPERTY(int, age)
	JSON_PROPERTY(QList<QString>, hobbies)
};
#endif // !PERSON_TEST_H

#ifndef TEST_PAGE_INFO_H
#define TEST_PAGE_INFO_H

#include "JsonSerializer.h"

class TestPageInfo final: public JsonSerializable
{
	Q_GADGET
	JSON_SERIALIZABLE
public:
	JSON_PROPERTY(int, totalNumber)
	JSON_PROPERTY(int, totalPage)
	JSON_PROPERTY(int, pageSize)
	JSON_PROPERTY(int, currentPage)
};
#endif // !TEST_PAGE_INFO_H

#ifndef TEST_PAGED_PERSON_H
#define TEST_PAGED_PERSON_H

#include "JsonSerializer.h"
#include "TestPageInfo.h"
#include "TestPerson.h"

class TestPagedPerson final: public JsonSerializable
{
	Q_GADGET
	JSON_SERIALIZABLE
public:
	JSON_PROPERTY(TestPageInfo, page)
	//JSON_PROPERTY(QVector<TestPerson>, persons)
	JSON_PROPERTY(std::vector<TestPerson>, persons)
};
#endif // !TEST_PAGED_PERSON_H
TestPagedPerson pagedPerson;
TestPageInfo page;
page.set_totalNumber(80);
page.set_totalPage(4);
page.set_currentPage(1);
page.set_pageSize(80 / 4);
pagedPerson.set_page(page);
TestPerson person1;
person1.set_age(18);
person1.set_name("A");
person1.set_hobbies({ "running", "TV" });
TestPerson person2;
person2.set_age(16);
person2.set_name("B");
person2.set_hobbies({ "reading", "swimming" });
TestPerson person3;
person3.set_age(21);
person3.set_name("C");
person3.set_hobbies({ "gaming", "swimming" });
pagedPerson.set_persons({ person1, person2, person3 });
auto rawJson = pagedPerson.toRawJson();
qDebug().noquote() << rawJson;

TestPagedPerson newPagedPerson;
newPagedPerson.fromJson(rawJson);
posted @ 2024-09-29 15:13  非法关键字  阅读(114)  评论(0编辑  收藏  举报