STL容器之vector
【1】模板类vector
模板类vector可理解为广义数组。广义数组,即与类型无关的数组,具有与数组相同的所有操作。
那么,你或许要问:既然C++语言本身已提供了一个序列式容器array,为什么还要vector呢?
我们一直强调,事物总是朝向更先进的方向螺旋式发展与进化。这个问题其中的道理也不例外。
vector的数据安排及操作方式与array非常相似。两者唯一区别在于空间运用的灵活性。
array是静态空间,一旦配置不能改变。要想换个大(或小)点的房子,一切琐细得由客户端自己料理(步骤如下):
首先,配置一块新空间,然后从旧地址将元素逐一搬往新地址,最后把原来的空间释放还给系统。
vector是动态空间,不仅可以在运行期再确定容器容量大小,而且随着元素的加入,它的内部机制会自动扩充空间以容纳新元素。
所以,相比较array,运用vector对内存的合理灵活性利用有很大的帮助。
【2】vector应用实例
(1)有构造函数才会有容器。模板类vector的构造函数及构建实例。代码如下:
1 #include <iostream> 2 #include <vector> 3 #include <memory> 4 using namespace std; 5 6 void main() 7 { 8 /* C++11 9 default (1) 10 explicit vector (const allocator_type& alloc = allocator_type()); 11 // 构建一个空容器(默认构造函数)。 12 13 fill (2) 14 explicit vector (size_type n); 15 vector (size_type n, const value_type& val, 16 const allocator_type& alloc = allocator_type()); 17 // 构建一个容器。容器存储n个元素,每个元素的初始值为val(如果提供)。 18 19 range (3) 20 template <class InputIterator> 21 vector (InputIterator first, InputIterator last, 22 const allocator_type& alloc = allocator_type()); 23 // 构建一个容器。容器存储元素与源容器中[first,last)区间的元素内容一样。 24 // 内容指存放元素的值及位置和[first,last)区间中完全一样,方向也要一致。 25 26 copy (4) 27 vector (const vector& x); 28 vector (const vector& x, const allocator_type& alloc); 29 // 拷贝构造函数。构建一个容器,逐个复制源容器x中的每一个元素,存放到本容器中,方向也保持一致。 30 31 move (5) 32 vector (vector&& x); 33 vector (vector&& x, const allocator_type& alloc); 34 // 移动构造函数。如果新的分配器alloc和x的分配器alloc不一样,那么将移动源容器x里面的元素到新的vector。 35 // 如果分配器是一样的,那么将直接转换其所有权。 36 // 将右值引用中容器x所有元素的所有权移动到新的vector中,避免付出昂贵的复制代价。 37 38 initializer list (6) 39 vector (initializer_list<value_type> il, 40 const allocator_type& alloc = allocator_type()); 41 // 初始化列表构造函数。从初始化列表il中复制每一个元素,放到新的vector中,方向与列表保持一致。 42 */ 43 // default (1) 44 vector<int> vInt1; 45 46 // fill (2) 47 int n; 48 cin >> n; 49 vector<int> vInt2(n); 50 51 vector<int> vInt3(10); 52 53 vector<int> vInt4(10, 100); 54 55 // range(3) 56 vector<int> vInt5(vInt4.begin(), vInt4.end()); 57 58 int myInt[] = {1, 2, 3, 4, 5}; 59 vector<int> vInt6(myInt, myInt + sizeof(myInt) / sizeof(int)); 60 61 // copy(4) 62 vector<int> vInt7(vInt4); 63 64 // move(5) 65 vector<unique_ptr<string> > vUS = vector<unique_ptr<string> >({ 66 unique_ptr<string> (new string("Qin")), 67 unique_ptr<string> (new string("Liu")), 68 unique_ptr<string> (new string("Wang")), 69 unique_ptr<string> (new string("Zhang"))}); 70 71 // initializer_list(6) 72 vector<int> vInt8({10, 20, 30, 40, 50}); 73 }
对于任何一个类而言,构造函数是相当重要的,其所有对象呈现的操作方法都与构造函数息息相关。
类的构造函数为对象先天性的行为铺垫(比如:类的构造函数若多一个参数,那么类将会有很多成员方法为这个参数服务)。
下面的实例演示使用vector模板方便创建动态分配数组的优势(内置数组无法完成)
1 #include <iostream> 2 #include <vector> 3 #include <string> 4 using namespace std; 5 6 void main() 7 { 8 cout << "请输入预处理学生数目:\n"; 9 int nStudents; 10 cin >> nStudents; 11 vector<string> names(nStudents); 12 vector<int> scores(nStudents); 13 cout << "请输入" << nStudents << " 个学生的姓名和英语成绩 (0 ~ 100)。\n"; 14 int i; 15 for (i = 0; i < nStudents; ++i) 16 { 17 cout << "姓名 No" << i + 1 << ": "; 18 cin >> names[i]; 19 cout << "成绩 (0 ~ 100):"; 20 cin >> scores[i]; 21 cin.get(); 22 } 23 cout << "输入所有学生信息结束。\n"; 24 cout << "打印输入姓名及成绩如下:\n"; 25 for (i = 0; i < nStudents; ++i) 26 { 27 cout << names[i] << "\t" << scores[i] << endl; 28 } 29 30 cin.get(); 31 } 32 // Run out: 33 /* 34 请输入预处理学生数目: 35 5 36 请输入5 个学生的姓名和英语成绩 (0 ~ 100)。 37 姓名 No1: sun 38 成绩 (0 ~ 100):89 39 姓名 No2: li 40 成绩 (0 ~ 100):87 41 姓名 No3: wang 42 成绩 (0 ~ 100):90 43 姓名 No4: qin 44 成绩 (0 ~ 100):98 45 姓名 No5: zhang 46 成绩 (0 ~ 100):69 47 输入所有学生信息结束。 48 打印输入姓名及成绩如下: 49 sun 89 50 li 87 51 wang 90 52 qin 98 53 zhang 69 54 */
vector模板更多的方法请看下节。
(2)常用成员方法应用示例如下代码:
1 #include <iostream> 2 #include <vector> 3 using namespace std; 4 5 void printVectorValue(vector<int> & vInt) 6 { 7 for (vector<int>::iterator pt = vInt.begin(); pt != vInt.end(); ++pt) 8 { 9 cout << *pt << "\t"; 10 } 11 cout << endl; 12 } 13 14 void main() 15 { 16 // vector对象的常用方法以及应用示例如下: 17 cout << "动态构建一个整型容器(请输入整数值,0时退出):\n"; 18 vector<int> vMyInts; 19 int nTemp; 20 while (cin >> nTemp && nTemp != 0) 21 { 22 vMyInts.push_back(nTemp); 23 } 24 cout << "您输入 " << vMyInts.size() << "个元素,分别如下: " << endl; 25 printVectorValue(vMyInts); 26 cout << "构建整型容器vMyInts完成" << endl << endl; 27 28 //(1)a.assign(b.begin(), b.begin() + 3); // b为向量,将b的0~2个元素构成的向量赋给a 29 cout << "(1) assign方法 :" << endl; 30 vector<int> vA1; 31 vA1.assign(vMyInts.begin(), vMyInts.begin() + 3); 32 cout << "容器vA1元素如下:" << endl; 33 printVectorValue(vA1); 34 cout << "(1) assign方法应用示例完成。" << endl << endl; 35 36 //(2)a.assign(4, 2); // 容器a只含4个元素,且每个元素值为2 37 cout << "(2) assign(4, 2)方法 :" << endl; 38 vector<int> vA2; 39 vA2.assign(4, 12); 40 cout << "容器vA2元素如下:" << endl; 41 printVectorValue(vA2); 42 cout << "(2) assign(4, 2)方法应用示例完成。" << endl << endl; 43 44 //(3)a.back(); // 返回容器a的最后一个元素 45 cout << "(3) back方法 :" << endl; 46 cout << "vMyInts容器的最后一个元素是:" << vMyInts.back() << endl; 47 cout << "(3) back方法应用示例完成。" << endl << endl; 48 49 // (4)a.front(); // 返回容器a的第一个元素 50 cout << "(4) front方法 :" << endl; 51 cout << "vMyInts容器的最前一个元素是:" << vMyInts.front() << endl; 52 cout << "(4) front方法应用示例完成。" << endl << endl; 53 54 // (5)a[i]; // 返回a的第i个元素,当且仅当a[i]存在 55 cout << "(5) a[i] 方法 :" << endl; 56 cout << "vMyInts容器的第2个元素是:" << vMyInts[2 - 1] << endl; 57 cout << "(5) a[i]方法应用示例完成。" << endl << endl; 58 59 // (6)a.clear(); // 清空容器a中的元素 60 cout << "(6) clear() 方法 :" << endl; 61 cout << "清空前,vA2容器中的元素为: " << endl; 62 printVectorValue(vA2); 63 vA2.clear(); 64 cout << "清空后,vA2容器中的元素为: " << endl; 65 printVectorValue(vA2); 66 cout << "(6) clear()方法应用示例完成。" << endl << endl; 67 68 // (7)a.empty(); // 判断容器a是否为空,空则返回ture,不空则返回false 69 cout << "(7) empty() 方法 :" << endl; 70 cout << "vA2容器是否为空:" << vA2.empty() << endl; 71 cout << "(7) empty()方法应用示例完成。" << endl << endl; 72 73 // (8)a.pop_back(); // 删除容器a的最后一个元素 74 cout << "(8) pop_back() 方法 :" << endl; vA2.clear(); 75 cout << "删除前,vMyInts容器中的元素为: " << endl; 76 printVectorValue(vMyInts); 77 vMyInts.pop_back(); 78 cout << "删除最后一个元素后,vMyInts容器中的元素为: " << endl; 79 printVectorValue(vMyInts); 80 cout << "(8) pop_back()方法应用示例完成。" << endl << endl; 81 82 // (9)a.erase(a.begin() + 1, a.begin() + 3); 83 // 删除容器a中第1个(从第0个算起)到第2个元素,也就是说删除的元素从[a.begin() + 1, a.begin() + 3)(前闭后开区间) 84 cout << "(9) erase() 方法 :" << endl; vA2.clear(); 85 cout << "删除前,vMyInts容器中的元素为: " << endl; 86 printVectorValue(vMyInts); 87 vMyInts.erase(vMyInts.begin() + 1, vMyInts.begin() + 3); 88 cout << "删除后,vMyInts容器中的元素为: " << endl; 89 printVectorValue(vMyInts); 90 cout << "(9) erase()方法应用示例完成。" << endl << endl; 91 92 // (10)a.push_back(5); // 在容器a的最后一个向量后插入一个元素,其值为5 93 cout << "(10) push_back() 方法 :" << endl; vA2.clear(); 94 cout << "添加前,vMyInts容器中的元素为: " << endl; 95 printVectorValue(vMyInts); 96 vMyInts.push_back(99); 97 cout << "添加后,vMyInts容器中的元素为: " << endl; 98 printVectorValue(vMyInts); 99 cout << "(10) push_back()方法应用示例完成。" << endl << endl; 100 101 // (11)a.insert(a.begin() + 1, 5); 102 // 在容器a的第1个元素(从第0个算起)的位置插入数值5,如a为1,2,3,4 插入元素后为1,5,2,3,4 103 cout << "(11) insert() 方法 :" << endl; 104 cout << "插入前,vMyInts容器中的元素为: " << endl; 105 printVectorValue(vMyInts); 106 vMyInts.insert(vMyInts.begin() + 1, 11); 107 cout << "插入后,vMyInts容器中的元素为: " << endl; 108 printVectorValue(vMyInts); 109 cout << "(11) insert()方法应用示例完成。" << endl << endl; 110 111 // (12)a.insert(a.begin() + 1, 3, 5); // 在容器a的第1个元素(从第0个算起)的位置插入3个数,其值都为5 112 cout << "(12) insert() 方法 :" << endl; 113 cout << "插入前,vMyInts容器中的元素为: " << endl; 114 printVectorValue(vMyInts); 115 vMyInts.insert(vMyInts.begin() + 1, 2, 12); 116 cout << "插入后,vMyInts容器中的元素为: " << endl; 117 printVectorValue(vMyInts); 118 cout << "(12) insert()方法应用示例完成。" << endl << endl; 119 120 // (13)a.insert(a.begin() + 1, b + 3, b + 6); 121 // b为数组。在a的第1个元素(从第0个算起)的位置插入b的第3个元素到第5个元素(不包括b + 6) 122 // b为1,2,3,4,5,9,8 插入元素后为1,4,5,9,2,3,4,5,9,8 123 cout << "(13) insert() 方法 (容器示例):" << endl; 124 cout << "插入前,vMyInts容器中的元素为: " << endl; 125 printVectorValue(vMyInts); 126 cout << "插入前,vA1容器中的元素为: " << endl; 127 printVectorValue(vA1); 128 vA1.insert(vA1.begin() + 1, vMyInts.begin() + 3, vMyInts.begin() + 6); 129 cout << "插入后,vMyInts容器中的元素为: " << endl; 130 printVectorValue(vMyInts); 131 cout << "插入后,vA1容器中的元素为: " << endl; 132 printVectorValue(vA1); 133 cout << "(13) insert()方法应用示例完成。" << endl << endl; 134 135 cout << "(13) insert() 方法 (数组示例):" << endl; 136 cout << "插入前,vA1容器中的元素为: " << endl; 137 printVectorValue(vA1); 138 cout << "数组中的元素为: {1, 2, 3, 4, 5, 6}" << endl; 139 int nArrB[] = {1, 2, 3, 4, 5, 6}; 140 vA1.insert(vA1.begin() + 1, nArrB + 1, nArrB + 4); 141 cout << "插入后,vA1容器中的元素为: " << endl; 142 printVectorValue(vA1); 143 cout << "(13) insert()方法应用示例完成。" << endl << endl; 144 145 // (14)a.size(); // 返回容器a中元素的个数; 146 cout << "(14) size() 方法: " << endl; 147 cout << "vA1容器中的元素为: " << endl; 148 printVectorValue(vA1); 149 cout << "vA1容器中元素个数为:" << vA1.size() << endl; 150 cout << "(14) size()方法应用示例完成。" << endl << endl; 151 152 // (15)a.capacity(); // 返回容器a的容量 153 cout << "(15) capacity() 方法: " << endl; 154 cout << "vA1容器的容量为:" << vA1.capacity() << endl; 155 cout << "(15) capacity()方法应用示例完成。" << endl << endl; 156 157 // (16)a.rezize(10); // 将a的现有元素个数调至10个,多则删,少则补,其值随机 158 cout << "(16) resize() 方法: " << endl; 159 cout << "重新分配大小前,vA1容器中元素为: " << endl; 160 printVectorValue(vA1); 161 vA1.resize(6); 162 cout << "vA1.resize(6) 重新分配大小后,vA1容器中元素为:" << endl; 163 printVectorValue(vA1); 164 cout << "(16) resize()方法应用示例完成。" << endl << endl; 165 166 // (17)a.rezize(10, 2); // 将容器a的现有元素个数调至10个,多则删,少则补其值为2 167 cout << "(17) resize() 方法: " << endl; 168 cout << "重新分配大小前,vA1容器中元素为: " << endl; 169 printVectorValue(vA1); 170 vA1.resize(10, 100); 171 cout << "vA1.resize(10, 110) 重新分配大小后,vA1容器中元素为:" << endl; 172 printVectorValue(vA1); 173 cout << "(17) resize()方法应用示例完成。" << endl << endl; 174 175 // (18)a.reserve(100); 176 // 将容器a的容量(capacity)扩充至100,也就是说现在测试 a.capacity();的时候返回值是100。 177 // 这种操作只有在需要给a添加大量数据的时候才有意义, 178 // 因为这将避免内存多次容量扩充操作(当a的容量不足时电脑会自动扩容,当然这必然降低性能。) 179 cout << "(18) reserve() 方法: " << endl; 180 cout << "扩容前,vA1容器的容量为:" << vA1.capacity() << endl; 181 vA1.reserve(30); 182 cout << "扩容后,vA1容器的容量为:" << vA1.capacity() << endl; 183 cout << "(18) reserve()方法应用示例完成。" << endl << endl; 184 185 // (19)a.swap(b); // b为容器,将a中的元素和b中的元素进行整体性交换 186 cout << "(19) swap() 方法: " << endl; 187 cout << "vA1容器中的元素为: " << endl; 188 printVectorValue(vA1); 189 cout << "vMyInts容器中的元素为: " << endl; 190 printVectorValue(vMyInts); 191 vA1.resize(5); 192 vMyInts.resize(5); 193 vA1.swap(vMyInts); 194 cout << "vA1容器中的元素为: " << endl; 195 printVectorValue(vA1); 196 cout << "vMyInts容器中的元素为: " << endl; 197 printVectorValue(vMyInts); 198 cout << "(19) swap() 方法应用示例完成。 " << endl << endl; 199 200 // (20)a = b; // b为容器,将一个对象赋给另一个对象 201 cout << "(20) 赋值方法: " << endl; 202 cout << "vA1容器中的元素为: " << endl; 203 printVectorValue(vA1); 204 vector<int> vTemp = vA1; 205 cout << " vTemp = vA1 后,vTemp 容器中的元素为: " << endl; 206 printVectorValue(vTemp); 207 cout << "(20) 赋值方法应用示例完成。 " << endl << endl; 208 209 // (20)a == b; // b为容器,向量的比较操作还有!=,>=,<=,>,< 210 cout << "(21) 比较操作: " << endl; 211 cout << "vA1容器中的元素为: " << endl; 212 printVectorValue(vA1); 213 cout << " vTemp 容器中的元素为: " << endl; 214 printVectorValue(vTemp); 215 cout << "vA1 == vTemp 比较:" << (vA1 == vTemp) << endl; 216 cout << "(21) 比较操作应用示例完成。 " << endl << endl; 217 cin >> nTemp; 218 } 219 // Run out 220 /* 221 动态构建一个整型容器(请输入整数值,0时退出): 222 12 223 23 224 34 225 45 226 56 227 67 228 78 229 89 230 90 231 0 232 您输入 9个元素,分别如下: 233 12 23 34 45 56 67 78 89 90 234 构建整型容器vMyInts完成 235 236 (1) assign方法 : 237 容器vA1元素如下: 238 12 23 34 239 (1) assign方法应用示例完成。 240 241 (2) assign(4, 2)方法 : 242 容器vA2元素如下: 243 12 12 12 12 244 (2) assign(4, 2)方法应用示例完成。 245 246 (3) back方法 : 247 vMyInts容器的最后一个元素是:90 248 (3) back方法应用示例完成。 249 250 (4) front方法 : 251 vMyInts容器的最前一个元素是:12 252 (4) front方法应用示例完成。 253 254 (5) a[i] 方法 : 255 vMyInts容器的第2个元素是:23 256 (5) a[i]方法应用示例完成。 257 258 (6) clear() 方法 : 259 清空前,vA2容器中的元素为: 260 12 12 12 12 261 清空后,vA2容器中的元素为: 262 263 (6) clear()方法应用示例完成。 264 265 (7) empty() 方法 : 266 vA2容器是否为空:1 267 (7) empty()方法应用示例完成。 268 269 (8) pop_back() 方法 : 270 删除前,vMyInts容器中的元素为: 271 12 23 34 45 56 67 78 89 90 272 删除最后一个元素后,vMyInts容器中的元素为: 273 12 23 34 45 56 67 78 89 274 (8) pop_back()方法应用示例完成。 275 276 (9) erase() 方法 : 277 删除前,vMyInts容器中的元素为: 278 12 23 34 45 56 67 78 89 279 删除后,vMyInts容器中的元素为: 280 12 45 56 67 78 89 281 (9) erase()方法应用示例完成。 282 283 (10) push_back() 方法 : 284 添加前,vMyInts容器中的元素为: 285 12 45 56 67 78 89 286 添加后,vMyInts容器中的元素为: 287 12 45 56 67 78 89 99 288 (10) push_back()方法应用示例完成。 289 290 (11) insert() 方法 : 291 插入前,vMyInts容器中的元素为: 292 12 45 56 67 78 89 99 293 插入后,vMyInts容器中的元素为: 294 12 11 45 56 67 78 89 99 295 (11) insert()方法应用示例完成。 296 297 (12) insert() 方法 : 298 插入前,vMyInts容器中的元素为: 299 12 11 45 56 67 78 89 99 300 插入后,vMyInts容器中的元素为: 301 12 12 12 11 45 56 67 78 89 99 302 303 (12) insert()方法应用示例完成。 304 305 (13) insert() 方法 (容器示例): 306 插入前,vMyInts容器中的元素为: 307 12 12 12 11 45 56 67 78 89 99 308 309 插入前,vA1容器中的元素为: 310 12 23 34 311 插入后,vMyInts容器中的元素为: 312 12 12 12 11 45 56 67 78 89 99 313 314 插入后,vA1容器中的元素为: 315 12 11 45 56 23 34 316 (13) insert()方法应用示例完成。 317 318 (13) insert() 方法 (数组示例): 319 插入前,vA1容器中的元素为: 320 12 11 45 56 23 34 321 数组中的元素为: {1, 2, 3, 4, 5, 6} 322 插入后,vA1容器中的元素为: 323 12 2 3 4 11 45 56 23 34 324 (13) insert()方法应用示例完成。 325 326 (14) size() 方法: 327 vA1容器中的元素为: 328 12 2 3 4 11 45 56 23 34 329 vA1容器中元素个数为:9 330 (14) size()方法应用示例完成。 331 332 (15) capacity() 方法: 333 vA1容器的容量为:9 334 (15) capacity()方法应用示例完成。 335 336 (16) resize() 方法: 337 重新分配大小前,vA1容器中元素为: 338 12 2 3 4 11 45 56 23 34 339 vA1.resize(6) 重新分配大小后,vA1容器中元素为: 340 12 2 3 4 11 45 341 (16) resize()方法应用示例完成。 342 343 (17) resize() 方法: 344 重新分配大小前,vA1容器中元素为: 345 12 2 3 4 11 45 346 vA1.resize(10, 110) 重新分配大小后,vA1容器中元素为: 347 12 2 3 4 11 45 100 100 100 100 348 349 (17) resize()方法应用示例完成。 350 351 (18) reserve() 方法: 352 扩容前,vA1容器的容量为:13 353 扩容后,vA1容器的容量为:30 354 (18) reserve()方法应用示例完成。 355 356 (19) swap() 方法: 357 vA1容器中的元素为: 358 12 2 3 4 11 45 100 100 100 100 359 360 vMyInts容器中的元素为: 361 12 12 12 11 45 56 67 78 89 99 362 363 vA1容器中的元素为: 364 12 12 12 11 45 365 vMyInts容器中的元素为: 366 12 2 3 4 11 367 (19) swap() 方法应用示例完成。 368 369 (20) 赋值方法: 370 vA1容器中的元素为: 371 12 12 12 11 45 372 vTemp = vA1 后,vTemp 容器中的元素为: 373 12 12 12 11 45 374 (20) 赋值方法应用示例完成。 375 376 (21) 比较操作: 377 vA1容器中的元素为: 378 12 12 12 11 45 379 vTemp 容器中的元素为: 380 12 12 12 11 45 381 vA1 == vTemp 比较:1 382 (21) 比较操作应用示例完成。 383 */
类的成员方法应用示例运行结果如下:
这么多成员方法,为什么没有sort排序方法呢?
(3)通用成员函数。上面常用方法那么多,怎么没有排序方法呢?这点得追溯于STL的设计理念(尽可能一劳永逸)。
STL从更广泛的角度定义了非成员函数来执行这些操作,即不是为每个类都定义sort()成员函数,而是定义一个适用于所有容器类的非成员函数sort()。
试想一个问题,假设有10个类,每个类都支持10种操作,若每个类都有自己的成员函数,则需要定义100(10*10)个成员函数。而采用STL方式,只需要定义10个非成员函数即可。
另外需要注意,已有执行相同任务的非成员函数,STL有时还会为类定义一个成员函数,因为对于有些操作来说,类特定算法的效率比通用算法高。
下面学习几个所谓的公用函数:for_each()、random_shuffle()、sort()。
3.1 for_each函数接受3个参数,前两个是定义容器中区间的迭代器,最后一个是指向函数的指针。可以用for_each函数来代替for循环。例如:
1 #include <iostream> 2 #include <vector> 3 #include <algorithm> // for_each 4 using namespace std; 5 6 void printVectorValue(int & rInt) 7 { 8 cout << rInt << "\t"; 9 } 10 11 void main() 12 { 13 cout << "动态构建一个整型容器(请输入整数值,0时退出):\n"; 14 vector<int> vMyInts; 15 int nTemp; 16 while (cin >> nTemp && nTemp != 0) 17 { 18 vMyInts.push_back(nTemp); 19 } 20 cout << "您输入 " << vMyInts.size() << " 个元素,分别如下: " << endl; 21 cout << "使用for循环打印容器元素如下:" << endl; 22 for (vector<int>::iterator pIt = vMyInts.begin(); pIt != vMyInts.end(); ++pIt) 23 { 24 cout << *pIt << "\t"; 25 } 26 cout << endl << "使用for_each打印容器元素如下:" << endl; 27 for_each(vMyInts.begin(), vMyInts.end(), printVectorValue); 28 cout << endl << "构建整型容器vMyInts完成" << endl << endl; 29 cin >> nTemp; 30 } 31 // Run out 32 /* 33 动态构建一个整型容器(请输入整数值,0时退出): 34 12 35 23 36 34 37 45 38 56 39 67 40 78 41 89 42 90 43 0 44 您输入 9 个元素,分别如下: 45 使用for循环打印容器元素如下: 46 12 23 34 45 56 67 78 89 90 47 使用for_each打印容器元素如下: 48 12 23 34 45 56 67 78 89 90 49 构建整型容器vMyInts完成 50 */
这样可以避免显式使用迭代器变量。
3.2 random_shffle函数接受两个指定区间的迭代器参数,并随机排列该区间中的元素。应用示例代码如下:
1 #include <iostream> 2 #include <vector> 3 #include <string> 4 #include <algorithm> // for_each 5 using namespace std; 6 7 template<class T> 8 void printVectorValue(T & rInt) 9 { 10 cout << rInt << "\t"; 11 } 12 13 void main() 14 { 15 cout << "动态构建一个整型容器(请输入整数值,0时退出):\n"; 16 vector<int> vMyInts; 17 int nTemp; 18 while (cin >> nTemp && nTemp != 0) 19 { 20 vMyInts.push_back(nTemp); 21 } 22 cout << "您输入 " << vMyInts.size() << " 个元素,分别如下: " << endl; 23 cout << "使用for_each打印排序前容器元素如下:" << endl; 24 for_each(vMyInts.begin(), vMyInts.end(), printVectorValue<int>); 25 random_shuffle(vMyInts.begin(), vMyInts.end()); 26 cout << endl << "使用for_each打印排序后容器元素如下:" << endl; 27 for_each(vMyInts.begin(), vMyInts.end(), printVectorValue<int>); 28 29 cout << endl; 30 vector<string> vStr; 31 vStr.push_back("sun"); 32 vStr.push_back("zhang"); 33 vStr.push_back("huang"); 34 vStr.push_back("liu"); 35 vStr.push_back("qin"); 36 cout << endl << "打印排序前容器元素如下:" << endl; 37 for (int j = 0; j < vStr.size(); ++j) 38 { 39 cout << vStr[j].c_str() << "\t"; 40 } 41 std::random_shuffle(vStr.begin(), vStr.end()); // 迭代器 42 cout << endl << "打印排序后容器元素如下:" << endl; 43 for (int j = 0; j < vStr.size(); ++j) 44 { 45 cout << vStr[j].c_str() << "\t"; 46 } 47 cout << endl; 48 system("pause"); 49 }
程序运行结果如下:
3.3 sort函数也要求容器支持随机访问,该函数有两个版本:
第一个版本接受两个定义区间的迭代器参数,并使用为存储在容器中的类型元素定义的<运算符。
备注:如果元素为用户自定义类型,必须定义能够处理该类型对象的operator<()函数),对区间中的元素进行操作。
第二个版本接受3个参数,前两个参数也是指定区间的迭代器,最后一个参数是指向要使用的函数的指针(而不是比较的operator<()运算符)。
sort函数的应用示例代码如下:
1 #include <iostream> 2 #include <vector> 3 #include <string> 4 #include <algorithm> // for_each 5 using namespace std; 6 7 template<class T> 8 void printVectorValue(T & rInt) 9 { 10 cout << rInt << "\t"; 11 } 12 13 struct student 14 { 15 string strName; 16 int nAge; 17 18 student(string sName = "liu", int nAge = 22) : strName(sName), nAge(nAge) 19 {} 20 }; 21 22 bool worseThan(const student & stuObject1, const student & stuObject2) 23 { 24 if (stuObject1.nAge > stuObject2.nAge) 25 { 26 return true; 27 } 28 else if (stuObject1.nAge == stuObject2.nAge 29 && stuObject1.strName > stuObject2.strName) 30 { 31 return true; 32 } 33 else 34 { 35 return false; 36 } 37 } 38 39 void printValue(const student & rStuObject) 40 { 41 cout << rStuObject.strName << " " << rStuObject.nAge << "\n"; 42 } 43 44 void main() 45 { 46 cout << "sort函数版本1测试程序:" << endl; 47 cout << "动态构建一个整型容器(请输入整数值,0时退出):\n"; 48 vector<int> vMyInts; 49 int nTemp; 50 while (cin >> nTemp && nTemp != 0) 51 { 52 vMyInts.push_back(nTemp); 53 } 54 cout << "您输入 " << vMyInts.size() << " 个元素,分别如下: " << endl; 55 cout << "使用for_each打印排序前容器元素如下:" << endl; 56 for_each(vMyInts.begin(), vMyInts.end(), printVectorValue<int>); 57 sort(vMyInts.begin(), vMyInts.end()); // 版本1 58 cout << endl << "使用for_each打印排序后容器元素如下:" << endl; 59 for_each(vMyInts.begin(), vMyInts.end(), printVectorValue<int>); 60 61 cout << endl << endl << "sort函数版本2测试程序:" << endl; 62 vector<student> vStudents; 63 vStudents.push_back(student("sun", 23)); 64 vStudents.push_back(student("zhang", 24)); 65 vStudents.push_back(student("huang", 22)); 66 vStudents.push_back(student("liu", 21)); 67 vStudents.push_back(student("qin", 22)); 68 cout << "使用for_each打印排序前容器元素如下:" << endl; 69 for_each(vStudents.begin(), vStudents.end(), printValue); 70 sort(vStudents.begin(), vStudents.end(), worseThan); // 版本2 71 cout << "使用for_each打印排序后容器元素如下:" << endl; 72 for_each(vStudents.begin(), vStudents.end(), printValue); 73 cout << endl; 74 system("pause"); 75 }
程序运行结果如下:
通用非成员函数结束。
【3】学习《STL源码剖析》中vector
(1)vector容器可以自动扩充内存空间,具体的扩充规则测试代码如下:
1 #include <vector> 2 #include <iostream> 3 #include <algorithm> 4 using namespace std; 5 6 template<class T> 7 void printVectorValue(T & rInt) 8 { 9 cout << rInt << " "; 10 } 11 12 template<class T> 13 void printSizeAndCapacity(vector<T> & oVec) 14 { 15 for_each(oVec.begin(), oVec.end(), printVectorValue<T>); 16 cout << endl << "size = " << oVec.size() << endl; 17 cout << "capacity = " << oVec.capacity() << endl << endl; 18 } 19 20 int main() 21 { 22 vector<int> iv(2, 9); 23 printSizeAndCapacity(iv); 24 for (int i = 0; i < 8; ++i) 25 { 26 iv.push_back(i + 1); 27 printSizeAndCapacity<int>(iv); 28 } 29 30 system("pause"); 31 }
vector容器示意图如下:
运行结果如下:
有一个遗留问题:如size不为零时,扩充后容量应该是之前的2倍,根据上例实际情况,不应该有奇数值的容量!求再探索,作此备录。
(2)vector迭代器和数据结构,如下图所示:
(3)vector部分源码如下:
1 #include<iostream> 2 #include<memory.h> 3 using namespace std; 4 5 // alloc是SGI STL的空间配置器 6 template <class T, class Alloc = alloc> 7 class vector 8 { 9 public: 10 // vector的嵌套类型定义,typedefs用于提供iterator_traits<I>支持 11 typedef T value_type; 12 typedef value_type* pointer; 13 typedef value_type* iterator; 14 typedef value_type& reference; 15 typedef size_t size_type; 16 typedef ptrdiff_t difference_type; 17 protected: 18 // 这个提供STL标准的allocator接口 19 typedef simple_alloc <value_type, Alloc> data_allocator; 20 21 iterator start; // 表示目前使用空间的头 22 iterator finish; // 表示目前使用空间的尾 23 iterator end_of_storage; // 表示实际分配内存空间的尾 24 25 void insert_aux(iterator position, const T& x); // 在position位置插入值为x的元素 26 27 // 释放分配的内存空间 28 void deallocate() 29 { 30 // 由于使用data_allocator进行内存空间的分配, 31 // 所以匹配使用data_allocator::deallocate()进行释放 32 // 如果直接释放, 对于data_allocator内部使用内存池的版本就会发生错误 33 if (start) 34 data_allocator::deallocate(start, end_of_storage - start); 35 } 36 37 void fill_initialize(size_type n, const T& value) 38 { 39 start = allocate_and_fill(n, value); 40 finish = start + n; // 设置当前使用内存空间的结束点 41 // 构造阶段(不多分配内存) 42 // 所以要设置内存空间结束点(与已经使用的内存空间结束点相同) 43 end_of_storage = finish; 44 } 45 46 public: 47 // 获取几种迭代器 48 iterator begin() { return start; } 49 iterator end() { return finish; } 50 51 // 返回当前对象个数 52 size_type size() const { return size_type(end() - begin()); } 53 size_type max_size() const { return size_type(-1) / sizeof(T); } 54 // 返回重新分配内存前最多能存储的对象个数 55 size_type capacity() const { return size_type(end_of_storage - begin()); } 56 // 判空 57 bool empty() const { return begin() == end(); } 58 // 重载[](下标访问) 59 reference operator[](size_type n) { return *(begin() + n); } 60 61 // 默认构造函数。默认构造vector不分配内存空间 62 vector() : start(0), finish(0), end_of_storage(0) {} 63 64 vector(size_type n, const T& value) { fill_initialize(n, value); } 65 vector(int n, const T& value) { fill_initialize(n, value); } 66 vector(long n, const T& value) { fill_initialize(n, value); } 67 68 // 需要对象提供默认构造函数 69 explicit vector(size_type n) { fill_initialize(n, T()); } 70 71 vector(const vector<T, Alloc>& x) 72 { 73 start = allocate_and_copy(x.end() - x.begin(), x.begin(), x.end()); 74 finish = start + (x.end() - x.begin()); 75 end_of_storage = finish; 76 } 77 78 ~vector() 79 { 80 // 析构对象 81 destroy(start, finish); 82 // 释放内存 83 deallocate(); 84 } 85 86 vector<T, Alloc>& operator=(const vector<T, Alloc>& x); 87 88 // 提供访问函数 89 reference front() { return *begin(); } 90 reference back() { return *(end() - 1); } 91 92 //////////////////////////////////////////////////////////////////////////////// 93 // 向容器尾追加一个元素, 可能导致内存重新分配(开辟新空间、移动原数据、释放原内存) 94 //////////////////////////////////////////////////////////////////////////////// 95 // push_back(const T& x) 96 // | 97 // |---------------- 容量已满? 98 // | 99 // ---------------------------- 100 // No | | Yes 101 // | | 102 // ↓ ↓ 103 // construct(finish, x); insert_aux(end(), x); 104 // ++finish; | 105 // |------ 内存不足, 重新分配 106 // | 大小为原来的2倍 107 // new_finish = data_allocator::allocate(len); <stl_alloc.h> 108 // uninitialized_copy(start, position, new_start); <stl_uninitialized.h> 109 // construct(new_finish, x); <stl_construct.h> 110 // ++new_finish; 111 // uninitialized_copy(position, finish, new_finish); <stl_uninitialized.h> 112 //////////////////////////////////////////////////////////////////////////////// 113 114 void push_back(const T& x) 115 { 116 // 内存满足条件则直接追加元素, 否则需要重新分配内存空间 117 if (finish != end_of_storage) 118 { 119 construct(finish, x); 120 ++finish; 121 } 122 else 123 insert_aux(end(), x); 124 } 125 126 127 //////////////////////////////////////////////////////////////////////////////// 128 // 在指定位置插入元素 129 //////////////////////////////////////////////////////////////////////////////// 130 // insert(iterator position, const T& x) 131 // | 132 // |------------ 容量是否足够 && 是否是end()? 133 // | 134 // ------------------------------------------- 135 // No | | Yes 136 // | | 137 // ↓ ↓ 138 // insert_aux(position, x); construct(finish, x); 139 // | ++finish; 140 // |-------- 容量是否够用? 141 // | 142 // -------------------------------------------------- 143 // Yes | | No 144 // | | 145 // ↓ | 146 // construct(finish, *(finish - 1)); | 147 // ++finish; | 148 // T x_copy = x; | 149 // copy_backward(position, finish - 2, finish - 1); | 150 // *position = x_copy; | 151 // ↓ 152 // data_allocator::allocate(len); <stl_alloc.h> 153 // uninitialized_copy(start, position, new_start); <stl_uninitialized.h> 154 // construct(new_finish, x); <stl_construct.h> 155 // ++new_finish; 156 // uninitialized_copy(position, finish, new_finish); <stl_uninitialized.h> 157 // destroy(begin(), end()); <stl_construct.h> 158 // deallocate(); 159 //////////////////////////////////////////////////////////////////////////////// 160 161 iterator insert(iterator position, const T& x) 162 { 163 size_type n = position - begin(); 164 if (finish != end_of_storage && position == end()) 165 { 166 construct(finish, x); 167 ++finish; 168 } 169 else 170 insert_aux(position, x); 171 return begin() + n; 172 } 173 174 iterator insert(iterator position) { return insert(position, T()); } 175 176 void pop_back() 177 { 178 --finish; 179 destroy(finish); 180 } 181 182 iterator erase(iterator position) 183 { 184 if (position + 1 != end()) 185 copy(position + 1, finish, position); 186 --finish; 187 destroy(finish); 188 return position; 189 } 190 191 192 iterator erase(iterator first, iterator last) 193 { 194 iterator i = copy(last, finish, first); 195 // 析构掉需要析构的元素 196 destroy(i, finish); 197 finish = finish - (last - first); 198 return first; 199 } 200 201 // 调整size, 但是并不会重新分配内存空间 202 void resize(size_type new_size, const T& x) 203 { 204 if (new_size < size()) 205 erase(begin() + new_size, end()); 206 else 207 insert(end(), new_size - size(), x); 208 } 209 void resize(size_type new_size) { resize(new_size, T()); } 210 211 void clear() { erase(begin(), end()); } 212 213 protected: 214 // 分配空间, 并且复制对象到分配的空间处 215 iterator allocate_and_fill(size_type n, const T& x) 216 { 217 iterator result = data_allocator::allocate(n); 218 uninitialized_fill_n(result, n, x); 219 return result; 220 } 221 222 // 提供插入操作 223 //////////////////////////////////////////////////////////////////////////////// 224 // insert_aux(iterator position, const T& x) 225 // | 226 // |---------------- 容量是否足够? 227 // ↓ 228 // ----------------------------------------- 229 // Yes | | No 230 // | | 231 // ↓ | 232 // 从opsition开始, 整体向后移动一个位置 | 233 // construct(finish, *(finish - 1)); | 234 // ++finish; | 235 // T x_copy = x; | 236 // copy_backward(position, finish - 2, finish - 1); | 237 // *position = x_copy; | 238 // ↓ 239 // data_allocator::allocate(len); 240 // uninitialized_copy(start, position, new_start); 241 // construct(new_finish, x); 242 // ++new_finish; 243 // uninitialized_copy(position, finish, new_finish); 244 // destroy(begin(), end()); 245 // deallocate(); 246 //////////////////////////////////////////////////////////////////////////////// 247 248 template <class T, class Alloc> 249 void insert_aux(iterator position, const T& x) 250 { 251 if (finish != end_of_storage) // 还有备用空间 252 { 253 // 在备用空间起始处构造一个元素,并以vector最后一个元素值为其初值 254 construct(finish, *(finish - 1)); 255 ++finish; 256 T x_copy = x; 257 copy_backward(position, finish - 2, finish - 1); 258 *position = x_copy; 259 } 260 else // 已无备用空间 261 { 262 const size_type old_size = size(); 263 const size_type len = old_size != 0 ? 2 * old_size : 1; 264 // 以上配置元素:如果大小为0,则配置1(个元素大小) 265 // 如果大小不为0,则配置原来大小的两倍 266 // 前半段用来放置原数据,后半段准备用来放置新数据 267 268 iterator new_start = data_allocator::allocate(len); // 实际配置 269 iterator new_finish = new_start; 270 // 将内存重新配置 271 try 272 { 273 // 将原vector的安插点以前的内容拷贝到新vector 274 new_finish = uninitialized_copy(start, position, new_start); 275 // 为新元素设定初值 x 276 construct(new_finish, x); 277 // 调整水位 278 ++new_finish; 279 // 将安插点以后的原内容也拷贝过来 280 new_finish = uninitialized_copy(position, finish, new_finish); 281 } 282 catch(...) 283 { 284 // 回滚操作 285 destroy(new_start, new_finish); 286 data_allocator::deallocate(new_start, len); 287 throw; 288 } 289 // 析构并释放原vector 290 destroy(begin(), end()); 291 deallocate(); 292 293 // 调整迭代器,指向新vector 294 start = new_start; 295 finish = new_finish; 296 end_of_storage = new_start + len; 297 } 298 } 299 300 //////////////////////////////////////////////////////////////////////////////// 301 // 在指定位置开始连续插入n个值为x的元素 302 //////////////////////////////////////////////////////////////////////////////// 303 // insert(iterator position, size_type n, const T& x) 304 // | 305 // |---------------- 插入元素个数是否为0? 306 // ↓ 307 // ----------------------------------------- 308 // No | | Yes 309 // | | 310 // | ↓ 311 // | return; 312 // |----------- 内存是否足够? 313 // | 314 // ------------------------------------------------- 315 // Yes | | No 316 // | | 317 // |------ (finish - position) > n? | 318 // | 分别调整指针 | 319 // ↓ | 320 // ---------------------------- | 321 // No | | Yes | 322 // | | | 323 // ↓ ↓ | 324 // 插入操作, 调整指针 插入操作, 调整指针 | 325 // ↓ 326 // data_allocator::allocate(len); 327 // new_finish = uninitialized_copy(start, position, new_start); 328 // new_finish = uninitialized_fill_n(new_finish, n, x); 329 // new_finish = uninitialized_copy(position, finish, new_finish); 330 // destroy(start, finish); 331 // deallocate(); 332 //////////////////////////////////////////////////////////////////////////////// 333 334 template <class T, class Alloc> 335 void insert(iterator position, size_type n, const T& x) 336 { 337 // 如果n为0,则不进行任何操作 338 if (n != 0) 339 { 340 if (size_type(end_of_storage - finish) >= n) 341 { // 剩下的备用空间大于等于“新增元素的个数” 342 T x_copy = x; 343 // 以下计算插入点之后的现有元素个数 344 const size_type elems_after = finish - position; 345 iterator old_finish = finish; 346 if (elems_after > n) 347 { 348 // 插入点之后的现有元素个数 大于 新增元素个数 349 uninitialized_copy(finish - n, finish, finish); 350 finish += n; // 将vector 尾端标记后移 351 copy_backward(position, old_finish - n, old_finish); 352 fill(position, position + n, x_copy); // 从插入点开始填入新值 353 } 354 else 355 { 356 // 插入点之后的现有元素个数 小于等于 新增元素个数 357 uninitialized_fill_n(finish, n - elems_after, x_copy); 358 finish += n - elems_after; 359 uninitialized_copy(position, old_finish, finish); 360 finish += elems_after; 361 fill(position, old_finish, x_copy); 362 } 363 } 364 else 365 { // 剩下的备用空间小于“新增元素个数”(那就必须配置额外的内存) 366 // 首先决定新长度:就长度的两倍 , 或旧长度+新增元素个数 367 const size_type old_size = size(); 368 const size_type len = old_size + max(old_size, n); 369 // 以下配置新的vector空间 370 iterator new_start = data_allocator::allocate(len); 371 iterator new_finish = new_start; 372 __STL_TRY 373 { 374 // 以下首先将旧的vector的插入点之前的元素复制到新空间 375 new_finish = uninitialized_copy(start, position, new_start); 376 // 以下再将新增元素(初值皆为n)填入新空间 377 new_finish = uninitialized_fill_n(new_finish, n, x); 378 // 以下再将旧vector的插入点之后的元素复制到新空间 379 new_finish = uninitialized_copy(position, finish, new_finish); 380 } 381 # ifdef __STL_USE_EXCEPTIONS 382 catch(...) 383 { 384 destroy(new_start, new_finish); 385 data_allocator::deallocate(new_start, len); 386 throw; 387 } 388 # endif /* __STL_USE_EXCEPTIONS */ 389 destroy(start, finish); 390 deallocate(); 391 start = new_start; 392 finish = new_finish; 393 end_of_storage = new_start + len; 394 } 395 } 396 } 397 };
(4)使用主要问题及成员函数详解。
4.1 erase函数。vector::erase()方法有两种重载形式:
(1.1)iterator erase (iterator _Where);
(1.2)iterator erase (iterator _First, iterator _Last);
如果是删除指定位置的元素时:返回值是一个迭代器,指向删除元素下一个元素。
如果是删除某范围内的元素时:返回值也表示一个迭代器,指向最后一个删除元素的下一个元素。
请看如下“临床典型性”应用错误示例代码:
1 #include <vector> 2 #include <iostream> 3 #include <algorithm> 4 using namespace std; 5 6 template<class T> 7 void printVectorValue(T & rInt) 8 { 9 cout << rInt << " "; 10 } 11 12 void main() 13 { 14 vector <int> v1; 15 for (int i = 0; i < 6; ++i) 16 { 17 v1.push_back(i + 10); 18 } 19 cout << "(1) v1 容器元素如下:" << endl; 20 for_each(v1.begin(), v1.end(), printVectorValue<int>); 21 cout << endl; 22 23 v1.erase(v1.begin( )); 24 cout << "(2) v1 容器元素如下:" << endl; 25 for_each(v1.begin(), v1.end(), printVectorValue<int>); 26 cout << endl; 27 28 v1.erase(v1.begin( ) + 1, v1.begin( ) + 3); 29 cout << "(3) v1 容器元素如下:" << endl; 30 for_each(v1.begin(), v1.end(), printVectorValue<int>); 31 cout << endl; 32 33 v1.push_back(14); 34 vector<int>::iterator itPt; 35 for (itPt = v1.begin(); itPt != v1.end(); ++itPt) 36 { 37 if (14 == *itPt) 38 { 39 v1.erase(itPt); 40 } 41 } 42 43 system("pause"); 44 }
程序运行结果如下:
分析程序:一一遍历容器找到元素值为14,然后一一删除。程序为什么会崩溃呢?
其实,出现这种原因是没搞懂erase的删除原理(套路),当调用erase()后迭代器就失效了,即变成一个野指针。
所以要处理这种问题,关键是要解决调用erase()方法后,迭代器变成野指针的问题。
结合套路,解决这个问题有两种方案:
第一种:正向删除。
第二种:逆向删除。代码如下:
1 #include <vector> 2 #include <iostream> 3 #include <algorithm> 4 using namespace std; 5 6 template<class T> 7 void printVectorValue(T & rInt) 8 { 9 cout << rInt << " "; 10 } 11 12 // 正删 13 template<class T> 14 void eraseAll(vector<T> & v1, T value) 15 { 16 vector<T>::iterator itPt; 17 for (itPt = v1.begin(); itPt != v1.end();) 18 { 19 if (value == *itPt) 20 { 21 itPt = v1.erase(itPt); 22 } 23 else 24 { 25 ++itPt; 26 } 27 } 28 } 29 30 // 逆删 31 template<class T> 32 void eraseAll2(vector<T> & v1, T value) 33 { 34 vector<T>::iterator itPt; 35 for (itPt = --v1.end(); itPt != v1.begin();) 36 { 37 if (value == *itPt) 38 { 39 v1.erase(itPt--); 40 } 41 else 42 { 43 --itPt; 44 } 45 } 46 47 if (value == *itPt) 48 { 49 v1.erase(itPt); 50 } 51 } 52 53 void main() 54 { 55 vector<int> v1(6, 9); 56 for (int i = 0; i < 6; ++i) 57 { 58 v1.push_back(i + 10); 59 } 60 cout << "(1) v1 容器元素如下:" << endl; 61 for_each(v1.begin(), v1.end(), printVectorValue<int>); 62 cout << endl; 63 64 v1.erase(v1.begin( )); 65 cout << "(2) v1 容器元素如下:" << endl; 66 for_each(v1.begin(), v1.end(), printVectorValue<int>); 67 cout << endl; 68 69 v1.erase(v1.begin( ) + 1, v1.begin( ) + 3); 70 cout << "(3) v1 容器元素如下:" << endl; 71 for_each(v1.begin(), v1.end(), printVectorValue<int>); 72 cout << endl; 73 74 v1.insert(v1.begin() + 5, 3, 12); 75 v1.push_back(15); 76 v1.push_back(15); 77 cout << "(4) 删除前 v1 容器元素如下:" << endl; 78 for_each(v1.begin(), v1.end(), printVectorValue<int>); 79 cout << endl; 80 81 // 第一种正向删除 82 eraseAll<int>(v1, 9); 83 eraseAll<int>(v1, 12); 84 eraseAll<int>(v1, 15); 85 86 /* 第二种逆向删除 87 eraseAll2<int>(v1, 9); 88 eraseAll2<int>(v1, 12); 89 eraseAll2<int>(v1, 15); 90 */ 91 cout << "(5) 删除9,12,15后 v1 容器元素如下:" << endl; 92 for_each(v1.begin(), v1.end(), printVectorValue<int>); 93 cout << endl; 94 /* 错误删除!! 95 for (itPt = v1.begin(); itPt != v1.end(); ++itPt) 96 { 97 if (14 == *itPt) 98 { 99 v1.erase(itPt); 100 } 101 } 102 */ 103 104 system("pause"); 105 } 106 107 // Run out 108 /* 109 (1) v1 容器元素如下: 110 9 9 9 9 9 9 10 11 12 13 14 15 111 (2) v1 容器元素如下: 112 9 9 9 9 9 10 11 12 13 14 15 113 (3) v1 容器元素如下: 114 9 9 9 10 11 12 13 14 15 115 (4) 删除前 v1 容器元素如下: 116 9 9 9 10 11 12 12 12 12 13 14 15 15 15 117 (5) 删除9,12,15后 v1 容器元素如下: 118 10 11 13 14 119 请按任意键继续. . . 120 */
规避开失效指针,完全可以正常删除。
为了加深理解,再追加一个正向删除的示例程序:
1 #include <vector> 2 #include <iostream> 3 4 #define PRINT_INFO (1) 5 6 void eraseTarget(std::vector<int>& vect, int value) 7 { 8 auto print = [&]() { 9 for (auto iter = vect.begin(); iter != vect.end();) 10 std::cout << (*iter++) << ((iter == vect.end()) ? std::string{} : " | "); 11 std::cout << std::endl; 12 }; 13 14 #if PRINT_INFO 15 std::cout << "************ Begin erase target ************" << std::endl; 16 std::cout << "begin element of vector :: "; 17 print(); 18 std::cout << std::endl; 19 #endif 20 21 auto iter = vect.begin(); 22 while (iter != vect.end()) 23 { 24 #if PRINT_INFO 25 print(); 26 std::cout << "before erase current iter value :: " << (*iter) << " || before erase size :: " << vect.size() << std::endl; 27 #endif 28 29 if ((*iter) == value) 30 { 31 iter = vect.erase(iter); 32 33 #if PRINT_INFO 34 std::cout << "after erase current iter value :: " << ((iter == vect.end()) ? (-1) : (*iter)) 35 << " || after erase size :: " << vect.size() << std::endl << std::endl; 36 #endif 37 } 38 else 39 { 40 #if PRINT_INFO 41 std::cout << std::endl; 42 #endif 43 ++iter; 44 } 45 } 46 47 #if PRINT_INFO 48 std::cout << "end element of vector :: "; 49 print(); 50 std::cout << "************ End erase target ************" << std::endl; 51 #endif 52 } 53 54 int main() 55 { 56 { 57 std::vector<int> vect = { 3, 3, 4, 5, 6, 3, 3, 7, 8, 9, 3, 3 }; 58 eraseTarget(vect, 3); 59 } 60 61 return 0; 62 }
备注:可通过宏PRINT_INFO控制是否打印过程信息,便于理解。
附本地运行结果:
4.2 insert函数
insert函数源码如下:
insert函数两种情况分析如下:
insert函数重载多,使用比较方便。
【4】vector总结
相比C++内置的array数组类,vector模板类是有生命的、动态的、可塑性的数组,使用更方便,更健壮。
Good Good Study, Day Day Up.
顺序 选择 循环 总结