QVariantMap和QVariant
typedef QVariantMap
Synonym for(同义词) <, >.
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
5 QMap<QString, QVariant> fruitMap;
6 fruitMap["Orange"] = 2.10;
7 fruitMap["Pineapple"] = 3.85;
8 fruitMap["Pear"] = pearMap;
QVariant被⽤于构建Qt Meta-Object,因此是QtCore的⼀部分。当然,我们也可以在GUI模块中使⽤,例如
QIcon icon("open.png");
QVariant variant = icon;
// other function
QIcon icon = variant.value<QIcon>();
我们使⽤了value()模版函数,获取存储在QVariant中的数据。这种函数在⾮GUI数据中同样适⽤,但是,在⾮GUI模块中,我们通常使⽤toInt()这样的⼀系列to...()函数,如toString()等。
  如果你觉得QVariant提供的存储数据类型太少,也可以⾃定义QVariant的存储类型。被QVariant存储的数据类型需要有⼀个默认的构造函数和⼀个拷贝构造函数。为了实现这个功能,⾸先必须使⽤Q_DECLARE_METATYPE()宏。通常会将这个宏放在类的声明所在头⽂件的下⾯():
1 要使⽤⼀个⾃定义类型可⽤于QVariant中只需要在类声明的后⾯加上:Q_DECLARE_METATYPE(),
union是什么类型2 struct MyClass
3 {
4 QString name;
5 int age;
6 }
7 Q_DECLARE_METATYPE(MyClass)
8
9
10 这样我们的类就可以像QMetaType::Type类⼀样使⽤没什么不同,有点不同的是使⽤⽅法上⾯只能这样使⽤.
11 MyClass myClass;
12 QVariant v3 = QVairant::fromValue(myClass);
13 //
14 v2.canConvert<MyClass>();
15 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");
转⾃:blog.csdn/zhangbinsijifeng/article/details/50686753
在使⽤VC、Delphi编写⽤户界⾯程序的时候,经常会把对象与控件的data域进⾏绑定,便于程序运⾏中读写提⾼效率。然⽽在Qt编程中怎么实现这个功能呢?⽐如将⼀个⽤户⾃定义的结构体与QComboBox控件的data绑定?
Qt的联机帮助做的实在是差强⼈意,⽰例代码太少了。调试了半天,看到给了我提⽰。主要步骤如下:
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
8 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);
转⾃:blog.csdn/derryzhang/article/details/5114491