Qt 模板库详细介绍是本文要介绍的内容吧,QT4引入了一组名为郁金香(Tulip)的容器类,用来取代老的 QCollection(QT2.3)基于指针(pointer-based)的 容器类和QTL(QT3)基于值(value-based)的容器类。
- Qt3模板库(QTL) 是一套提供对象容器的模板。请看Qt模板库
- QMap 提供基于值的一个字典的模板类
- QMapConstIterator QMap的常量迭代器
- QMapIterator QMap的迭代器
- QPair 提供基于值的一对元素的模板类
- QValueList 提供基于值的一个双向链表的模板类
- QValueListConstIterator QValueList的常量迭代器
- QValueListIterator QValueList的迭代器
- QValueStack 提供基于值的一个堆栈模板类
- QValueVector 提供基于值的一个动态数组模板类
Qt模板库(QTL)是一套提供对象容器的模板。如果你的编译器不能用到一个适当的STL实现,QTL可以替代它。它提供了对象的列表、对象的矢量(Vector 动态数组)、从一种类型到其它的映射(或字典),并且有关联的迭代器(iterator)和算法。一个容器是可以包含和管理其它对象的一个对象并且提供迭代器来允许被包含的对象能够被访问。
QTL类的命名约定与其他Qt类一致(比如,count()、isEmpty())。它们也提供额外的函数来兼容STL算法,比如size()和empty()。程序员也可以像使用STL的map一样来使用它们。
与STL相比,QTL仅仅包含了STL容器应用程序接口的最重要的特性,没有平台差异,通常要慢一些并且经常扩展为更少的对象代码。
如果你不能复制你所想要存储的对象,你***使用QPtrCollection和它的朋友。它们就是被设计用来正确地处理这些类型的指针语义。这将适用于比如所有继承QObject的类。QObject没有一个复制构造函数,所以把它们作为值来使用是不可能的。你也许可以选择存储QObject的指针到QValueList,但是直接使用QPtrList看起来是对这类应用程序领域的更好的选择。QPtrList,像所有其它的基于QPtrCollection的容器,提供了比一个速度优化了的基于值的容器更多健全的检查。
如果你有一些实现值语义的对象,并且在你的目标平台没有可用的STL,Qt模板库就可以替代它。值语义至少需要以下这些:
一个复制构造函数,
一个赋值操作符和
一个默认构造函数,比如一个没有任何参数的构造函数。
注意一个快速的复制构造函数对于容器的总性能是完全至关重要的,因为许多复制操作将会发生。
如果你打算排序你的数据,那么你必须在你的数据类中实现operator<()。
基于值的类的候选对象有QRect、QPoint、QSize、QString和所有简单的C++类型,比如int、bool和double。
Qt模板库是因为速度而被设计。迭代器是非常快的。为了实现这样的性能,所以就比基于QPtrCollection的类做了更少的错误检查。一个QTL容器,比如,没有跟踪任何关联的迭代器。这样就在比如删除项目的时候没有自动地执行有效性检查,但无论如何,它提供了很快很好的性能。
- operator<()
自然地,这些排序模板在常量迭代器下不能工作。
- QString second( "Einstein" );
- QString name( "Albert" );
- qSwap( second, name );
- QValueList<int> l;
- l.push_back( 1 );
- l.push_back( 1 );
- l.push_back( 1 );
- l.push_back( 2 );
- int c = 0;
- qCount( l.begin(), l.end(), 1, c ); // c == 3
- QValueList<int> l;
- l.push_back( 1 );
- l.push_back( 1 );
- l.push_back( 1 );
- l.push_back( 2 );
- QValueListIterator<int> it = qFind( l.begin(), l.end(), 2 );
- QValueVector<int> v(3);
- qFill( v.begin(), v.end(), 99 ); // v包含99, 99, 99
- QValueVector<int> v1(3);
- v1[0] = 1;
- v1[2] = 2;
- v1[3] = 3;
- QValueVector<int> v2(5);
- v1[0] = 1;
- v1[2] = 2;
- v1[3] = 3;
- v1[4] = 4;
- v1[5] = 5;
- bool b = qEqual( v1.begin(), v2.end(), v2.begin() );
- // b == TRUE
- QValueList<int> l;
- l.push_back( 100 );
- l.push_back( 200 );
- l.push_back( 300 );
- QTextOStream str( stdout );
- qCopy( l.begin(), l.end(), QTextOStreamIterator(str) );
- QValueVector<int> vec(3);
- vec.push_back( 100 );
- vec.push_back( 200 );
- vec.push_back( 300 );
- QValueVector<int> another;
- qCopyBackward( vec.begin(), vec.end(), another.begin() );
- // “another”现在包含100、200、300
- // 无论如何元素都被一次性复制
- // 是按倒序排列的(300、200、100)
另外,你可以把任何一个Qt模板库的迭代器作为OutputIterator使用。只需要注意迭代器的右面现在存在的元素和你所想要插入的一样多。下面这个例子就说明了这些:
- QStringList l1, l2;
- l1 << "Weis" << "Ettrich" << "Arnt" << "Sue";
- l2 << "Torben" << "Matthias";
- qCopy( l2.begin(), l2.end(), l1.begin() );
- QValueVector<QString> v( l1.size(), "Dave" );
- qCopy( l2.begin(), l2.end(), v.begin() );
这段代码结束后,列表l1包含“Torben”、“Matthias”、“Arnt”和“Sue”,前面的内容被覆盖了。矢量v包含“Torben”、“Matthias”、“Dave”和“Dave”,也是前面的内容被覆盖了。
如果你写了新的算法,请考虑把它们写成模板函数,这样就可以使它们能够用在尽可能多的容器上了。在上一个例子中,你可以很容易地使用qCopy()打印出一个标准C++数组:
- int arr[] = { 100, 200, 300 };
- QTextOStream str( stdout );
- qCopy( arr, arr + 3, QTextOStreamIterator( str ) );流
- QDataStream str(...);
- QValueList<QRect> l;
- // ……在这里填充这个列表
- str << l;
容易可以这样被再一次地读入:
- QValueList<QRect> l;
- str >> l;
这些也同样适用于QStringList、QValueStack和QMap。
小结:Qt 模板库详细介绍的内容就介绍到这里,希望本文对你有帮助!