解析 Qt 模板庫 詳細介紹
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 模板庫詳細介紹的內容就介紹到這里,希望本文對你有幫助!