Qt QMetaType自定义类型和QVariant的序列化

时间:2011-04-20 08:02:53

标签: c++ qt

我希望能够访问QSettings和QVariant使用的序列化技术。例如,如果您创建一个QRect对象并将其存储到带有QSettings的INI文件中,则会得到如下所示的行:

value=@Rect(1 2 3 4)

大多数标准Qt类型和自定义类型都有类似的字符串序列化格式用于保存/加载。我怎么能做同样的事情?

也就是说,我有一个QVariant并希望保存它包含的数据,然后再加载该数据。保存的表单应该是文本数据(如上所述),可以在普通配置文件(如INI)或注册表中使用。


嗯,我查看了QSettings源代码,它只是对一些常见类型进行了硬编码处理,然后使用了QDataStream。这意味着没有通用的方法来以文本形式序列化数据。

3 个答案:

答案 0 :(得分:0)

QDataStream类提供二进制数据到QIODevice的序列化。 您应该实现两个运算符:

QDataStream & operator<< ( QDataStream & stream, const YourClass & yourObject ); QDataStream & operator>> ( QDataStream & stream, YourClass & yourObject );

将负责数据的序列化和反序列化。

Read more about serialization in Qt

如果您对文本序列化感兴趣,那么您应该选择QTextStream 作为你的工具。但是,大多数类都没有能够处理文本流的操作符,因此您必须实现它们。

答案 1 :(得分:0)

Qt 元对象系统能够为自定义类型注册大量运算符。其中之一是StreamOperator。 QSettings 使用此运算符来 在配置文件中写入和读取 QVariant。

因此,首先,您需要为@divanov 提到的自定义类型实现两个流运算符

QDataStream & operator<< ( QDataStream & stream, const YourClass & yourObject );
QDataStream & operator>> ( QDataStream & stream, YourClass & yourObject );

之后,需要为自定义类型注册这两个操作符才能 使用 qRegisterMetaTypeStreamOperators 的 Qt 元对象系统。

以下示例描述了前面提到的所有步骤 自定义类型颜色。

#include <QMetaType>
#include <QDataStream>
#include <QSettings>
#include <cassert>

// Custom type 'Color'
struct Color
{
  uint8_t _red;
  uint8_t _green;
  uint8_t _blue;

  // Stream operator used by QSettings to save a value of type Color 
  // to configuration file 
  friend QDataStream& operator<<(QDataStream& out, const Color& color)
  {
      out << color._red;
      out << color._green;
      out << color._blue;
      return out;
  }

  // Stream operator used by QSettings to load a value of type Color 
  // from a configuration file
  friend QDataStream& operator>>(QDataStream& in, Color& color)
  {
    in >> color._red;
    in >> color._green;
    in >> color._blue;
    return in;
  }
};

Q_DECLARE_METATYPE( Color )

int main(int argc, char* argv[])
{
  Q_UNUSED(argc)
  Q_UNUSED(argv)

  // Register Color to qt meta-object system
  qRegisterMetaType<Color>();
 
  // Register its two streams operator to qt meta-object system
  qRegisterMetaTypeStreamOperators<Color>();

  // Test it with QSettings!
  QSettings configFile("config.ini");

  // Save the color
  Color saveColor { 12,  13,  14 };
  configFile.setValue("Color", QVariant::fromValue(saveColor));

  // Load the color
  Color loadColor = configFile.value("Color", QVariant()).value<Color>();

  // Asserts are successful
  assert(loadColor._red == 12);
  assert(loadColor._green == 13);
  assert(loadColor._blue == 14);
}

答案 2 :(得分:0)

我个人觉得 QVariantMapQVariantList 对这类事情非常方便。 为您的类/结构提供转换函数:

class User {
public:
    QVariantMap toVariantMap() const {
        QVariantMap map;
        map["name"] = m_name;
        map["reputation"] = m_reputation;
        map["tags"] = m_tags;

        return map;
    }

    static User fromVariantMap(const QVariantMap& map) {
        User user;
        user.m_name = map["name"].toString();
        user.m_reputation = map["reputation"].toInt();
        user.m_tags = map["tags"].toStringList();
        return user;
    }

private:
    QString m_name;
    int m_reputation;
    QStringList m_tags;
} 

toVariantMap 保存:

settings->setValue("user", user.toVariantMap());

使用 fromVariantMap 获取:

auto user = User::fromVariantMap(settings->value("user").toVariantMap());

要保存 QString 以外的项目列表,可以使用 QVariantList:

QVariantList list;
for (int i = 0; i < m_list.size(); ++i)
    list.append(m_list[i]);
map["list"] = list;