QVariant类及QVariant与自定义数据类型转换的方法

    这个类型相当于是Java里面的Object,它把绝大多数Qt提供的数据类型都封装起来,起到一个数据类型“擦除”的作用。比如我们的 table单元格可以是string,也可以是int,也可以是一个颜色值,那么这么多类型怎么返回呢?于是,Qt提供了这个QVariant类型,你可以把这很多类型都存放进去,到需要使用的时候使用一系列的to函数取出来即可。比如你把int包装成一个QVariant,使用的时候要用 QVariant::toInt()重新取出来。这里需要注意的是,QVariant类型的放入和取出必须是相对应的,你放入一个int就必须按int取出,不能用toString(), Qt不会帮你自动转换。

   数据核心无非就是一个 union,和一个标记类型的type:传递的是整数 123,那么它union存储整数123,同时type标志Int;如果传递字符串,union存储字符串的指针,同时type标志QString。

  QVariant 属于 Qt 的Core模块,属于Qt的底层核心之一,ActiveQt、QtScript、QtDeclarative等都严重依赖于QVariant。

  QVariant 可以保存很多Qt的数据类型,包括QBrush、QColor、QCursor、QDateTime、QFont、QKeySequence、 QPalette、QPen、QPixmap、QPoint、QRect、QRegion、QSize和QString,并且还有C++基本类型,如 int、float等。QVariant还能保存很多集合类型,如QMap<QSTRING, QVariant>, QStringList和QList。item view classes,数据库模块和QSettings都大量使用了QVariant类,,以方便我们读写数据。

  QVariant也可以进行嵌套存储,例如


 
 
  1. QMap<QString, QVariant> pearMap;
  2. pearMap[ "Standard"] = 1.95;
  3. pearMap[ "Organic"] = 2.25;
  4. QMap<QString, QVariant> fruitMap;
  5. fruitMap[ "Orange"] = 2.10;
  6. fruitMap[ "Pineapple"] = 3.85;
  7. fruitMap[ "Pear"] = pearMap;

   QVariant被用于构建Qt Meta-Object,因此是QtCore的一部分。当然,我们也可以在GUI模块中使用,例如


 
 
  1. QIcon icon("open.png");
  2. QVariant variant = icon;
  3. // other function
  4. QIcon icon = variant.value<QIcon>();

   我们使用了value()模版函数,获取存储在QVariant中的数据。这种函数在非GUI数据中同样适用,但是,在非GUI模块中,我们通常使用toInt()这样的一系列to...()函数,如toString()等。

   如果你觉得QVariant提供的存储数据类型太少,也可以自定义QVariant的存储类型。被QVariant存储的数据类型需要有一个默认的构造函数和一个拷贝构造函数。为了实现这个功能,首先必须使用Q_DECLARE_METATYPE()宏。通常会将这个宏放在类的声明所在头文件的下面(Q_DECLARE_METATYPE(MyClass)宏的位置:头文件,类声明后):


 
 
  1. 要使用一个自定义类型可用于QVariant中只需要在类声明的后面加上:Q_DECLARE_METATYPE(),
  2. struct MyClass
  3. {
  4. QString name;
  5. int age;
  6. }
  7. Q_DECLARE_METATYPE(MyClass)
  8. 这样我们的类就可以像QMetaType::Type类一样使用没什么不同,有点不同的是使用方法上面只能这样使用.
  9. MyClass myClass;
  10. QVariant v3 = QVairant::fromValue(myClass);
  11. //
  12. v2.canConvert<MyClass>();
  13. MyClass myClass2 = v2.value<MyClass>();

   例如:

     Q_DECLARE_METATYPE(BusinessCard)

  然后我们就可以使用:


 
 
  1. BusinessCard businessCard;
  2. QVariant variant = QVariant::fromValue(businessCard);
  3. // ...
  4. if (variant.canConvert<BusinessCard>()) {
  5. BusinessCard card = variant.value<BusinessCard>();
  6. // ...
  7. }

    由于VC 6的编译器限制,这些模板函数不能使用,如果你使用这个编译器,需要使用qVariantFromValue(), qVariantValue()和qVariantCanConvert()这三个宏。

  如果自定义数据类型重写了<<和>>运算符,那么就可以直接在QDataStream中使用。不过首先需要使用qRegisterMetaTypeStreamOperators().宏进行注册。这就能够让QSettings使用操作符对数据进行操作,例如qRegisterMetaTypeStreamOperators<BusinessCard>("BusinessCard");

 

QVariant与自定义数据类型转换的方法

     1、将自定义数据类型使用Q_DECLARE_METATYPE宏进行声明,便于编译器识别。

     2、在插入对象的时候,声明QVariant对象,使用其setValue方法添加自定义对象。

     3、在读取的时候,采用QVariant对象的value方法返回自定义对象。

 示例代码如下:

    1.自定义结构体及元类型声明:


 
 
  1. struct MyStruct
  2. {
  3. int a;
  4. char b[ 10];
  5. };
  6. MyStruct stu;
  7. Q_DECLARE_METATYPE(MyStruct)

   2.在程序初始化中,给结构体初始化并存储到QComboBox的data域中:


 
 
  1. bzero(&stu, sizeof(MyStruct)); //stu就是上面声明的全局变量
  2. //赋初值
  3. stu.a = 100;
  4. strcpy(stu.b, "Hello./n");
  5. //类型转换
  6. QVariant v;
  7. QString ss = QString( "%1").arg(stu.b);
  8. v.setValue(stu);
  9. //保存到控件data中
  10. ui->cboTest->addItem(ss, v);
  11. ui->cboTest->addItem( "aadkjf", 0);

   3.读取的时候反过来处理,如下:


 
 
  1. QVariant v;
  2. //从控件data域取得variant对象
  3. v = ui->cboTest->itemData( 0);
  4. //转换为自定义的结构体实例
  5. MyStruct s = v.value<MyStruct>();
  6. printf( "value=%d:%s/n",s.a, s.b);

参考:

https://www.yuque.com/docs/share/03a4fa64-7a63-48a3-9592-3f7f83aec61f

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值