【Qt笔记】使用 QJson 处理 JSON

XML 曾经是各类应用的配置和传输的首选方式。可是如今 XML 遇到了一个强劲的对手:JSON。咱们能够在这里看到有关 JSON 的语法。整体来讲,JSON 的数据比 XML 更紧凑,在传输效率上也要优于 XML。不过 JSON 数据的层次化表达不及 XML,至少不如 XML 那样突出。不过这并不会阻止 JSON 的普遍应用。python

 

一个典型的 JSON 文档能够像下面的例子:c++

{
  "encoding" : "UTF-8",
  "plug-ins" : [
      "python",
      "c++",
      "ruby"
      ],
  "indent" : { "length" : 3, "use_space" : true }
}

JSON 的全称是 JavaScript Object Notation,与 JavaScript 密不可分。熟悉 JavaScript 的童鞋立刻就会发现,JSON 的语法就是 JavaScript 对象声明的语法。JSON 文档其实就是一个 JavaScript 对象,于是也称为 JSON 对象,以大括号做为起止符,其实质是不排序的键值对,其中键要求是 string 类型,值能够是任意类型。好比上面的示例,键 encoding 的值是字符串 UTF-8;键 plug-ins 的值是一个数组类型,在 JSON 中,数组以中括号表示,这个数组是一个字符串列表,分别有 python、c++ 和 ruby 三个对象;键 indent 的值是一个对象,这个对象有两个属性,length = 3,use_space = true。git

对于 JSON 的解析,咱们可使用 QJson 这个第三方库。QJson 能够将 JSON 数据转换为 QVariant 对象,将 JSON 数组转换成 QVariantList 对象,将 JSON 对象转换成 QVariantMap 对象。咱们在这里使用 git clone 出 QJson 的整个代码。注意 QJson 没有提供连接库的 pro 文件,所以咱们只须要将全部源代码文件添加到咱们的项目便可(如同这些文件是咱们本身写的同样)。接下来就可使用 QJson 读取 JSON 内容:github

#include "parser.h"
//////////
QJson::Parser parser;
bool ok;

QString json("{"
        "\"encoding\" : \"UTF-8\","
        "\"plug-ins\" : ["
        "\"python\","
        "\"c++\","
        "\"ruby\""
        "],"
        "\"indent\" : { \"length\" : 3, \"use_space\" : true }"
        "}");
QVariantMap result = parser.parse(json.toUtf8(), &ok).toMap();
if (!ok) {
    qFatal("An error occurred during parsing");
    exit (1);
}

qDebug() << "encoding:" << result["encoding"].toString();
qDebug() << "plugins:";

foreach (QVariant plugin, result["plug-ins"].toList()) {
    qDebug() << "\t-" << plugin.toString();
}

QVariantMap nestedMap = result["indent"].toMap();
qDebug() << "length:" << nestedMap["length"].toInt();
qDebug() << "use_space:" << nestedMap["use_space"].toBool();

将 JSON 对象转换成QVariant对象很简单,基本只须要下面几行:json

// 1. 建立 QJson::Parser 对象
QJson::Parser parser;

bool ok;
// 2. 将 JSON 对象保存在一个对象 json 中,进行数据转换
QVariant result = parser.parse (json, &ok);

QJson::Parser::parse()函数接受两个参数,第一个参数是 JSON 对象,能够是QIODevice *或者是QByteArray;第二个参数是转换成功与否,若是成功则被设置为 true。函数返回转换后的QVariant对象。注意咱们转换后的对象实际上是一个QVariantMap类型,能够像QMap同样使用重载的 [] 获取键所对应的值。另外,因为 result[“plug-ins”] 是一个QVariantList对象(由于是由 JSON 数组返回的),于是能够调用其toList()函数,经过遍历输出每个值。数组

若是须要将QVariant生成 JSON 对象,咱们则使用QJson::Serializer对象。例如:ruby

QVariantList people;

QVariantMap bob;
bob.insert("Name", "Bob");
bob.insert("Phonenumber", 123);

QVariantMap alice;
alice.insert("Name", "Alice");
alice.insert("Phonenumber", 321);

people << bob << alice;

QJson::Serializer serializer;
bool ok;
QByteArray json = serializer.serialize(people, &ok);

if (ok) {
    qDebug() << json;
} else {
    qCritical() << "Something went wrong:" << serializer.errorMessage();
}

QJson::Serializer和前面的QJson::Parser的用法类似,只须要调用QJson::Serializer::serialize()便可将QVariant类型的数据转换为 JSON 格式。其返回值是QByteArray类型,能够用于不少其它场合。函数

上面是 QJson 的主要使用方法。其实 QJson 还提供了另一个类QObjectHelper,用于QVariantQObject之间的转换。注意咱们上面所说的 QJson 的转换须要的是QVariant类型的数据,不管是转换到 JSON 仍是从 JSON 转换而来。可是一般咱们在应用程序中使用的是QObject及其子类。QObjectHelper提供了一个工具函数,完成QVariantQObject之间的转换。例如咱们有下面的类:工具

class Person : public QObject
{
  Q_OBJECT

  Q_PROPERTY(QString name READ name WRITE setName)
  Q_PROPERTY(int phoneNumber READ phoneNumber WRITE setPhoneNumber)
  Q_PROPERTY(Gender gender READ gender WRITE setGender)
  Q_PROPERTY(QDate brithday READ brithday WRITE setBrithday)
  Q_ENUMS(Gender)

  public:
    Person(QObject* parent = 0);
    ~Person();

    QString name() const;
    void setName(const QString& name);

    int phoneNumber() const;
    void setPhoneNumber(const int phoneNumber);

    enum Gender {Male, Female};
    void setGender(Gender gender);
    Gender gender() const;

    QDate brithday() const;
    void setBrithday(const QDate& dob);

  private:
    QString m_name;
    int m_phoneNumber;
    Gender m_gender;
    QDate m_dob;
};

那么,咱们可使用下面的代码将Person类进行 JSON 序列化:spa

Person person;
person.setName("Flavio");
person.setPhoneNumber(123456);
person.setGender(Person::Male);
person.setDob(QDate(1982, 7, 12));

QVariantMap variant = QObjectHelper::qobject2qvariant(&person);
QJson::Serializer serializer;
qDebug() << serializer.serialize( variant);

以及:

QJson::Parser parser;
QVariant variant = parser.parse(json);
Person person;
QObjectHelper::qvariant2qobject(variant.toMap(), &person);
相关文章
相关标签/搜索