欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

关于QVariant 的学习一点分享

程序员文章站 2024-03-17 18:06:34
...

关于QVariant 的学习一点分享

本人就像铁匠铺里的学徒,一点一滴地积累Qt开发的一些技巧。看到一个Demo程序中用到了QVariant 进行传值。就像搞明白它。分析了网上大牛们的介绍,于是,自己也想拙笔写一篇。

Variant中文翻译为“变体,转化”。

这个数据类型的出现,有什么意义呢?

那么还是查阅的Qt官方的开发手册吧!

Variant类的开发手册解读

1、因为C++ 是禁止使用 Unions 联合体的,多以大多数的Qt的来不能被以联合体的数据类型来使用。

2、一个Qvariant 对象可以是某个数据类型,或者是很拥有很多值的数据,比如a string list(字符串列表)。

3、Variant 的数据类型很灵活,你可以find out(查明)该对象是何种数据类型,然后转化成另外一个类型。

举例如下:


    QVariant v(123);                // The variant now contains an int
    int x = v.toInt();              // x = 123
    qDebug() << v;                  // Writes a type tag and an int to out
    v = QVariant("hello");          // The variant now contains a QByteArray

    int y = v.toInt();              // y = 0 since v cannot be converted to an int
    QString s = v.toString();       // s = tr("hello")  (see QObject::tr())
    qDebug() << v;                 
    qDebug() << s;
    qDebug() << QString ::number(y);

输出:
关于QVariant 的学习一点分享

从上面的例子中可以看到, QVariant v 可以store 各种数据类型。不然如何被叫做“变体”的变量呢?

4、当然,QVariant 还支持复杂类型的数据类型。如QList、QMap<QString, QVariant> 、支持结构体等等。所以功能非常强大。

举例:

    QVariant v;
    QStringList sl;  //创建一个字符串列表
       sl<<"A"<<"B"<<"C"<<"D";
       v = QVariant(sl);  //将该列表保存在一个QVariant变量中
       //QVariant::type()函数返回存储在QVariant变量中的值的数据类型。
       if(v.type() == QVariant::StringList)
       {
           QStringList list=v.toStringList();
           for(int i=0;i<list.size();++i)
           {
               qDebug()<<list.at(i);  //输出列表内容
           }
       }

输出:
关于QVariant 的学习一点分享

结构体应用举例:

先构造一个结构,并声明

struct myStruct //自定义的数据类型
{
    int age;
    char name[10];
};
Q_DECLARE_METATYPE(myStruct)  //注册,必不可少

然后使用QVariant

   int id=qRegisterMetaType<MyStruct>();
           MyStruct stu;
           stu.age = 100;//也可以先定义变量后这样赋值
           strcpy(stu.name,"Hello./n");

           v.setValue(stu); //设置QVariant的值
           qDebug()<<"v:"<<v;

           MyStruct ste; //这部分代码主要是将QVariant类再转化为myStruct类,其他QVariant类转化成其他类也可用这种方法
           ste=v.value<MyStruct>();
           qDebug()<<"ste:"<<ste.age<<ste.name;

           if(v.canConvert(id)){
                          qDebug()<<"it can be converted by this means!";
                          MyStruct tempData=v.value<MyStruct>();
                           qDebug()<<"tempData:"<<tempData.age<<tempData.name;
                      }

输出:
关于QVariant 的学习一点分享
5、QVariant 也支持Null values(空值)

  QVariant x, y(QString()), z(QString(""));
  x.convert(QVariant::Int);
  // x.isNull() == true
  // y.isNull() == true, z.isNull() == false

到这里,对QVariant 的出现,它的用法基本有所了解了,它非常便捷。

当我们要理解一个新的事物时,用生活中常见的事物作类比,会帮助我们加深对这个新事物的理解程度和准确度。

这里参考一位技术大牛的介绍:

*在世界上还没有发明集装箱之前,所有的货物都是直接通过卡车运送至港口,再由港口的装卸工搬运至货船,由于所有的货物大小形状规格不同,导致装至船上的货物远远小于一艘船的实际承载量。而一次远航,就需要花费很长的周期和高昂的开销。对于海运的货物就变得非常昂贵。
而在出现集装箱之后,工厂生产的产品可以直接装在标准的集装箱中,所有的集装箱可以平整的放在运输船上,一艘船便可以运送大量的货物,从而产品价格低廉,增加了各个国家的经济来往,这也改变了世界的经济格局。
那么,你有没有想到,为什么这么简单的设计就改变了世界呢。这源自于集装箱屏蔽了每种产品的大小形状上的差异性,对外提供了一致的形状,那么所有的周边产品,例如港口的起重机,货船的设计都可以依据集装箱的标准而进行有效的设计。
好了,现在我们回到QVariant类(类比集装箱的特点),它正是屏蔽了不同类型的数据结构之间的差异性,从而可以让数据以一种标准的形式在类之间,函数之间,对象之间进行传递,就像集装箱让产品在不同国家之间进行传递一样,那么我们就可以依据QVariant提供的标准型,在其之上建立起足够通用便捷的代码。
例如,当一个函数使用QVariant作为参数时,这恰好就提供了这样的便捷性。就拿我们上篇文章的例子setProperty,因为第二个参数value要为所有的类型提供通用转换方法,那么使用QVariant作为这些类型的存储变量,那是再合适不过了。
QObject::setProperty(const char name, const QVariant &value)
当我们想要再获取该变量时,就使用property接口,用QVariant提供的toT方法进行转换,从而得到我们想要的值。这是一个多么便捷的做法啊,我还真惊叹于实现者的思维方式。从某种意义上讲,这也可能是多态的另一种表现方式。

QVariant 类的支持的类型:

QVariant支持多种数据类型的转换,包括C++的所有基本类型,Qt提供的基本类型,甚至还可以支持自定义类型的转换

下图是其支持的类型(仅列出了一部分,详细类型可以查看帮助文档或头文件声明)
关于QVariant 的学习一点分享

常用函数:

QVariant::canConvert(int targetTypeId) const

如果变量的类型可以转换为请求的类型targetTypeId,则返回true。

bool QVariant::convert(int targetTypeId)

将变量转换为请求的类型targetTypeId。如果转换不能完成,则清除变量。如果成功转换了变量的当前类型,则返回true;否则返回false。

T QVariant::value() const

返回转换为模板类型T的存储值。如果不能转换该值,将返回一个默认构造的值。

static QVariant QVariant::fromValue(const T &value)

返回一个包含值副本的QVariant。否则,其行为与setValue()完全相同。

在看下面一个例子:

QVariant 变量在类之间的传递:

#include <QVariant>
#include <QPoint>
#include <QDebug>

class MyClass
{
public:
    MyClass():i(0){}
    friend QDebug operator<<(QDebug d, const MyClass &c);
    
private:
    int i;
};
//注意,如果想要自定义的MyStruct也支持QVariant,那么需要Q_DECLARE_METATYPE向Qt元对象系统进行注册
Q_DECLARE_METATYPE(MyClass)
QDebug operator<<(QDebug d, const MyClass &c)
{
    d << "MyStruct(" << c.i << ")";
}

int main(int argc, char *argv[])
{
    QVariant v = 42;
    
    qDebug() << "v.canConvert<int>(): " << v.canConvert<int>() << v.toInt();
    qDebug() << "v.canConvert<QString>()" << v.canConvert<QString>() << v.toString();
    qDebug() << "v.canConvert<QPoint>()" << v.canConvert<QPoint>() << v.toPoint();
    
    qDebug() << "v covert to QString before:" << v;
    v.convert(QVariant::String);
    qDebug() << "v covert to QString after:" << v;
    v.convert(QVariant::Point);//如果无法转换,则QVariant本身会被清空
    qDebug() << "v covert to QPoint after:" << v;
    
    //支持自定义类型
    MyClass c;
    v.setValue(c);
    //...
    MyClass c2 = v.value<MyClass>();
    qDebug() << c2;
}

综上,QVariant可以在函数之间进行传递,同样的,我们也可以在类之间,甚至网络之间进行传递,而对象序列化也是需要QVariant这样的特性。

在今后的设计中,即使我们不使用QVariant,但是这样的思维模式是需要我们来学习的,屏蔽不同物体之间的差异,提供标准化的接口,然后依据该标准建造自己的框架体系

相关标签: Qt QVariant