Qt學習之容器的使用詳解
Qt容器
Qt容器主要優點就是在所有的平臺上的運行都表現的一致,並且它們都是隱含共享的.Qt容器的另外一個主要特征就是易於使用的迭代器類,它們可以利用QDataStream變成數據流,而且他們通常可以使用執行文件中的代碼量比相應的STL類中的要少.最後Qt/Embedded Linux 支持的一些硬件平臺隻能使用Qt容器.
連續容器
QVector<T>
向量與普通數組的區別在於,向量知道自己的大小,並且可以重新被定義大小,向量在末尾添加額外的項是非常快的.但是在向量前面或者中間插入項是非常耗時的.
初始化QVector
需要添加頭文件#include <QtCore/qvector.h>
// 初始化定義向量大小 QVector<double> qvect(3); qvect[0] = 0.1; qvect[1] = 0.2; qvect[2] = 0.3; // 初始化未定義向量 QVector<double> qvect1; qvect1.append(0.001); qvect1.append(0.002); qvect1.append(0.003);
當然,定義大小後也可以使用append函數再後面追加.
// 初始化定義向量大小 QVector<double> qvect2(3); qvect2.append(0.22); qvect2.append(0.21); qvect2.append(0.23); qvect2.append(0.24);
這樣追加後,向量的大小就會根據實際情況瞭.
QLinkedList<T>
QVector在開頭或者中間插入項或者除去這些項都是非常費時的,也可以使用QLinkedList,這是一種把項存儲在不相鄰的位置的數據結構.與QVector不同,QLinkedList(鏈表)不支持快速隨機訪問,但是它提供瞭"常量時間"的插入和刪除.QLinkedList未提供[ ]操作符,所以必須使用迭代器來遍歷.
需要添加頭文件#include <QtCore/qlinkedlist.h>
// QLinkedList 輸入輸出 QLinkedList<double> qlink; qlink.append(12); qlink.append(14); QLinkedList<double>::iterator i = qlink.begin(); for (; i != qlink.end(); ++i) { std::cout << " qlink = " << *i << std::endl; }
QList<T>
QList連續容器是一個數組列表,結合瞭QVector和QLinkedList的優點,支持隨機訪問,在任意一端插入或者刪除都非常快速.
需要添加頭文件#include <QtCore/qlist.h>
// Qlist QList<double> qlint; qlint.append(15); qlint.push_back(17); std::cout << "qlint = " << qlint[0] << std::endl; std::cout << "qlint = " << qlint[1] << std::endl;
註意的是,Qt的容易雖然和STL的不太一樣,但是Qt容器兼容STL的函數名,比如push_back這是STL容器的添加函數,但是Qt也是支持這個方法的,其實底層push_back就是調用append函數的.
QStringList
QStringList類是被廣泛應用於Qt應用編程接口的QList<QString>的子類.除瞭從它的基類繼承的函數以外,QStringList還提供瞭一些特別的函數.以使得這種類對字符串的處理方式更通用.
總結
對於所有的容器類,值類型T可以是一個與int, double,指針類型,具有默認構造函數的類(沒有參數的構造函數),拷貝構造函數或者賦值操作符相似的類.符合這個條件的類包括: QByteArray,QDateTime,QRegExp,QString和QVariant.
派生自QObject的Qt類不具備資格,因為他們沒有拷貝構造函數和賦值操作符.這在實際應用中並不是問題,因為可以簡單的存儲指向QObject類的指針而不是對象本身.
為瞭能使隱含共享的作用發揮到最好,可以采取兩個編程習慣
第一種:
對於一個向量或者列表進行讀取時采用at()函數,而不是[ ]操作符.因為Qt不能分辨 [ ]操作符是讀取還是賦值.而at() 函數則不被允許賦值.
第二種:
使用STL風格的迭代器遍歷容器的時候經可能的使用const_iterator, constBegin()和constEnd().
此外:Qt還提供瞭foreach循環.
// foreach 使用 foreach (double var , qlint) { std::cout << "foreach qlint = " << var << std::endl; } for each (double var in qlint) { std::cout << "STL foreach qlint = " << var << std::endl; } for (auto& var : qlint) { std::cout << "C++11 foreach qlint = " << var << std::endl; }
當然C++ 11 以後也支持for each 語法上稍有差距.習慣那個用哪個.Qt的foreach是Qt自己通過宏定義自己封裝的,而for each是C++11原生支持的,現在C++版本都支持,看自己習慣.
關聯容器
關聯容器主要是QMap<K,T>和QHash<K,T>
QMap<K,T>
需要添加頭文件#include <QtCore/qmap.h>和#include <QtCore/qhash.h>
在映射中插入項的一點簡單方式是調用insert():[ ]操作符既可以用於檢索也可以用於插入.如果在非常量映射中使用[ ]為一個不存在的鍵檢索值,則會給定的鍵和空值創建一個新的項.為瞭避免意外的創建空的值可以使用value()函數來代替[ ]操作符來獲取項.
// Qmap操作 QMap<QString, int> map; map.insert("123",123); map.insert("1",1); map.insert("2",2); map["3"] = 3; map["4"] = 4; int val = map.value("123"); std::cout << "map val = " << val << std::endl;
QMap有一對方便的函數,keys()和values()它們在處理小數據集時顯得特別有用.他們分別返回映射鍵和映射值.
QList<QString> strKey = map.keys(); QList<int> iValue = map.values(); for each (QString var in strKey) { std::cout << "map key = " << var.toStdString() << std::endl; } for (auto& var : iValue) { std::cout << "map value = " << var << std::endl; } foreach(QString var, strKey) { std::cout << "map key = " << var.toStdString() << std::endl; std::cout << "map value = " << map.value(var) << std::endl; }
var.toStdString()這個是QString轉string的函數,map.value(var)這個是獲取key的value的值.
通過insertMulti()函數或者QMultiMap<K,T>的子函數可以讓多個鍵值對有相同的鍵.
字符串,字節數組,變量操作
QString,QByteArray和QVariant 這三種類與容器類都有相似之處.有些情況都可以用來代替容器類.
C++本身提供兩種形式的字符串,一種是以'\0'結尾的字符數組和std::string類.與這兩種字符串不同,QString支持16位Unicode值.Unicode碼以ASCII碼和Latin-1碼為子集可以表示世界上對大多數的語言.
字符串拼接
// 字符串拼接 QString str; str = "123" + QString("qwe"); str += "asf"; str.append("zzxc"); std::cout << "str = " << str.toStdString() << std::endl; str.sprintf("%s %.1f","poi",123.4); std::cout << "str = " << str.toStdString() << std::endl; str = QString("%1 %2 (%3 - %4)").arg("rt").arg("gd").arg("vtr").arg("re");
數字轉字符串
QString nstr = QString::number(123);
字符串轉數字
auto a = nstr.toInt(); auto a = nstr.toDouble(); auto a = nstr.toLongLong();
字符串截取
QString astr = "polluter pays principle"; std::cout << "astr = " << astr.mid(9,4).toStdString() << std::endl; std::cout << "astr = " << astr.mid(9).toStdString() << std::endl; std::cout << "astr = " << astr.left(8).toStdString() << std::endl; std::cout << "astr = " << astr.right(9).toStdString() << std::endl;
輸出pays mid的參數第一個是從字符串第幾個開始截取,第二個參數是截取幾個字符,當然第二個參數沒有就是截取到最後.left()函數是從左邊截取幾個字符,參數就是截取的數量,right()函數也是一樣的,從右邊截取.
字符串查詢是否包含某個子字符串
QString bstr = "the middle bit"; std::cout << "pos = " << bstr.indexOf("middle") << std::endl; if (bstr.startsWith("the")) { } if (bstr.endsWith("bit")) { }
indexOf()函數查詢bstr是否包含"middle"子字符串.如果包含返回字符串位置,如果隻想檢查是否以某個字符串開始,以某個字符串結束,可以使用startsWith()和endsWith()
字符串替換
bstr.replace(0,3,"The"); std::cout << "bstr = " << bstr.toStdString() << std::endl; bstr.replace("the", "The"); std::cout << "bstr = " << bstr.toStdString() << std::endl;
replace()函數第一個參數是字符串從第幾個開始,第二個參數是結束位置,第三個是替換的字符串.
第二重載版本就比較簡單瞭,第一個參數是需要被替換的字符串,第二個是替換的字符串.兩個字符串不一定長度一樣,不一樣也可以.
字符串刪除多餘的空格
刪除字符串兩端的空格
QString cstr = " BOB \t THE \nDOG \n"; std::cout << "bstr = " << cstr.trimmed().toStdString() << std::endl;
trimmed()函數刪除字符串兩端的空格.
刪除字符串內部連續的空格
std::cout << "bstr = " << cstr.simplified().toStdString() << std::endl;
simplified()函數刪除字符串內部連續的空格.
把一個字符串分成一些QStringList字串
QString dstr = "polluter pays principle"; QStringList qlst = dstr.split(" "); std::cout << "qlst length = " << qlst.length() << std::endl; for each (QString var in qlst) { std::cout << "var = " << var.toStdString() << std::endl; }
split()函數把QString分成QStringList,參數就是按那個字符分,demo是按" "空格字符分的.
將QStringList拼接成QString
QString estr = qlst.join("\n"); std::cout << "estr = " << estr.toStdString() << std::endl;
join()函數把QStringList拼接成QString,中間參數是要加的字符,如果不加用""空的就行,我加瞭\n比較好看.
其他簡單的小函數
length() 函數返回字符串長度
isEmpty() 判斷字符串是否為空
實驗所有的代碼
實驗環境使用VS2015+qt5.9,使用Qt控制臺輸出的.實驗都可以輸出正確結果
#include <QtCore/QCoreApplication> #include <QtCore/qvector.h> #include <QtCore/qlinkedlist.h> #include <QtCore/qalgorithms.h> #include <QtCore/qmap.h> #include <QtCore/qhash.h> #include <QtCore/qlist.h> #include <iostream> int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); std::cout << "hello world" << std::endl; // 初始化定義向量大小 QVector<double> qvect(3); qvect[0] = 0.1; qvect[1] = 0.2; qvect[2] = 0.3; // 初始化未定義向量 QVector<double> qvect1; qvect1.append(0.001); qvect1.append(0.002); qvect1.append(0.003); // 初始化定義向量大小 QVector<double> qvect2(3); qvect2.append(0.22); qvect2.append(0.21); qvect2.append(0.23); qvect2.append(0.24); // QLinkedList 輸入輸出 QLinkedList<double> qlink; qlink.append(12); qlink.append(14); QLinkedList<double>::iterator i = qlink.begin(); for (; i != qlink.end(); ++i) { std::cout << " qlink = " << *i << std::endl; } // Qlist QList<double> qlint; qlint.append(15); qlint.push_back(17); std::cout << "qlint = " << qlint[0] << std::endl; std::cout << "qlint = " << qlint[1] << std::endl; // foreach 使用 foreach (double var , qlint) { std::cout << "foreach qlint = " << var << std::endl; } for each (double var in qlint) { std::cout << "STL foreach qlint = " << var << std::endl; } for (auto& var : qlint) { std::cout << "C++11 foreach qlint = " << var << std::endl; } // Qmap操作 QMap<QString, int> map; map.insert("123",123); map.insert("1",1); map.insert("2",2); map["3"] = 3; map["4"] = 4; int val = map.value("123"); std::cout << "map val = " << val << std::endl; QList<QString> strKey = map.keys(); QList<int> iValue = map.values(); for each (QString var in strKey) { std::cout << "map key = " << var.toStdString() << std::endl; } for (auto& var : iValue) { std::cout << "map value = " << var << std::endl; } foreach(QString var, strKey) { std::cout << "map key = " << var.toStdString() << std::endl; std::cout << "map value = " << map.value(var) << std::endl; } // 字符串拼接 QString str; str = "123" + QString("qwe"); str += "asf"; str.append("zzxc"); std::cout << "str = " << str.toStdString() << std::endl; str.sprintf("%s %.1f","poi",123.4); std::cout << "str = " << str.toStdString() << std::endl; str = QString("%1 %2 (%3 - %4)").arg("rt").arg("gd").arg("vtr").arg("re"); std::cout << "str = " << str.toStdString() << std::endl; QString nstr = QString::number(123); QString astr = "polluter pays principle"; std::cout << "astr = " << astr.mid(9,4).toStdString() << std::endl; std::cout << "astr = " << astr.mid(9).toStdString() << std::endl; std::cout << "astr = " << astr.left(8).toStdString() << std::endl; std::cout << "astr = " << astr.right(9).toStdString() << std::endl; QString bstr = "the middle bit"; std::cout << "pos = " << bstr.indexOf("middle") << std::endl; if (bstr.startsWith("the")) { } if (bstr.endsWith("bit")) { } bstr.replace(0,3,"The"); std::cout << "bstr = " << bstr.toStdString() << std::endl; bstr.replace("the", "The"); std::cout << "bstr = " << bstr.toStdString() << std::endl; QString cstr = " BOB \t THE \nDOG \n"; std::cout << "bstr = " << cstr.trimmed().toStdString() << std::endl; std::cout << "bstr = " << cstr.simplified().toStdString() << std::endl; QString dstr = "polluter pays principle"; QStringList qlst = dstr.split(" "); std::cout << "qlst length = " << qlst.length() << std::endl; for each (QString var in qlst) { std::cout << "var = " << var.toStdString() << std::endl; } QString estr = qlst.join("\n"); std::cout << "estr = " << estr.toStdString() << std::endl; return a.exec(); }
結果
到此這篇關於Qt學習之容器的使用詳解的文章就介紹到這瞭,更多相關Qt容器內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- C/C++ Qt 數據庫QSql增刪改查組件應用教程
- C/C++ Qt TreeWidget 嵌套節點操作使用
- Qt QTreeWidget 樹形結構實現代碼
- QT與javascript交互數據的實現
- C/C++ Qt QThread線程組件的具體使用