4196
备注:
|
14550
|
删除的内容标记成这样。 | 加入的内容标记成这样。 |
行号 7: | 行号 7: |
行号 9: | 行号 10: |
除此以外还有: * 特殊的容器:string(字符串), array(C语言原始数组) * 容器适配器:stack(栈), queue(队列), priority_queue(优先队列) |
除此以外还有: * 特殊的容器:string(字符串), array(C语言原始数组) * 容器适配器:stack(栈), queue(队列), priority_queue(优先队列) |
行号 13: | 行号 16: |
行号 17: | 行号 21: |
行号 21: | 行号 26: |
* 输出跌代子 | * 输出跌代子 |
行号 23: | 行号 29: |
行号 34: | 行号 41: |
{{{#!cplusplus | {{{ #!cplusplus |
行号 40: | 行号 49: |
行号 41: | 行号 51: |
行号 42: | 行号 53: |
{{{#!cplusplus | {{{ #!cplusplus |
行号 45: | 行号 57: |
vector < int > a4(a2); //构造一个vector与a2完全一样 | vector< int > a4(a2); //构造一个vector与a2完全一样 |
行号 48: | 行号 60: |
vector < int > a5( values, values+5); //通过迭代子来构造vector | vector< int > a5( values, values+5); //通过迭代子来构造vector |
行号 52: | 行号 64: |
{{{#!cplusplus | {{{ #!cplusplus |
行号 60: | 行号 73: |
行号 61: | 行号 75: |
{{{#!cplusplus | {{{ #!cplusplus |
行号 67: | 行号 82: |
行号 68: | 行号 84: |
{{{#!cplusplus | {{{ #!cplusplus |
行号 74: | 行号 91: |
行号 75: | 行号 93: |
{{{#!cplusplus | {{{ #!cplusplus |
行号 87: | 行号 106: |
== 用法实例 == {{{ #!cplusplus #include <vector> int main() { using namespace std; vector< string > sentence; sentence.reserve( 5 ); sentence.push_back( “ Hello, “); sentence.push_back( “how “); sentence.push_back( “are “); sentence.push_back( "you "); sentence.push_back( “?“); copy( sentence.begin(), sentence.end(), ostream_iterator<string>(cout, “ “)); cout << endl; cout << sentence.size() << endl; cout << sentence.capacity() << endl; swap( sentence[1], sentence[3]); sentence.insert( find(sentence.begin(), sentence.end(), “?”), “always”); sentence.back() = “!”; copy( sentence.rbegin(), sentence.rend(), ostream_iterator<string>(cout, “ “)); cout << endl; } }}} 用vector代替数组使用 {{{ #!cplusplus vector < int > a(10); // int a[10]; a[0] = 1; a[1] = 2; a[2] = a[0] + a[1]; for( int i = 0; i < a.size(); i++) scanf( “%d”, &a[i] ); }}} == vector的实现 == vector的实现类似于数据结构中的顺序表 attachment:vector.jpg {{{#!cplusplus template<class T, class Alloc=alloc> class vector { public: typedef T value_type; typedef value_type* iterator; typedef value_type& reference; typedef size_t size_type; protected: iterator start; iterator finish; iterator end_of_storage; public: iterator begin() { return start; } iterator end() { return finish; } size_type size() const { //返回当前元素个数 return size_type(end() - begin()); } bool empty() const { return begin() == end(); } reference operator[](size_type n) { return *(begin() + n); } reference front() { return *begin(); } reference back() { return *(end() - 1); } size_type capacity() const { //返回当前容量的大小 return size_type(end_of_storage - begin()); } size_type reserve(size_type n); //改变容量的大小 void push_back(const T& x) { if(finish != end_of_storage) { construct(finish, x); ++finish; } else { insert_aux(end(), x); } } protected: typedef simple_alloc<value_type, Alloc> data_allocator; void deallocate() { if(start) data_allocator::deallocate(start, end_of_storage - start); } void insert_aux(iterator position, const T& x) { if(finish != end_of_storage) { construct(finish, *(finish-1)); ++finish; T x_copy = x; copy_backward(position, finish-2, finish-1); *position = x_copy; } else { const size_type old_size = size(); const size_type len = old_size != 0 ? 2 * old_size : 1; iterator new_start = data_allocator::alloate(len); iterator new_finish = new_start; try { new_finish = uninitialized_copy(start, position, new_start); construct(new_finish, x); ++ new_finish; new_finish = uninitialized_copy(position, finish, new_finish); } catch(...) { destroy(new_start, new_finish); data_allocator::deallocate(new_start, len); throw; } destroy(begin(), end()); deallocate(); start = new_start; finish = new_finish; end_of_storage = new_start + len; } } }; }}} = deque双端队列 = deque与vector相似,区别是deque两端都是开放的,两端插入删除都很快。在头文件<deque>中定义。 attachment:deque.jpg 实现: attachment:deque_imp.jpg 可以随机访问,但速度比vector稍慢 迭代子是随机跌代子,是一种class而不是原始指针 操作非常相似,增加操作:push_front, pop_front,减少操作:reserve,capacity 任何插入和删除操作都可能使跌代子失效 例子: 例子: {{{#!cplusplus int main() { deque<string> coll; coll.assign (3, string("string")); coll.push_back ("last string"); coll.push_front ("first string"); copy (coll.begin(), coll.end(), ostream_iterator<string>(cout,"\n")); coll.pop_front(); coll.pop_back(); for (int i=1; i<coll.size(); ++i) { coll[i] = "another " + coll [i]; } coll.resize (4, "resized string"); copy (coll.begin(), coll.end(), ostream_iterator<string>(cout,"\n")); } }}} = list链表 = 一般为双链表实现 #include <list> 提供双向跌代子,不能随机访问 插入删除操作非常快速 插入删除操作不会使跌代子失效 提供了一些移动元素的算法,比通用算法更快 {{{#!cplusplus c1.swap(c2):交换两个链表的内容 c.remove(val) c.remove_if(predictor) c.unique() 删除重复元素 c.splice() 将一个链表中的元素切一部分到另一个链表 c.sort() 排序 c.merge() 合并两个链表 c.reverse() 倒置 }}} 例 {{{#!cplusplus void printLists (const list<int>& 11, const list<int>& 12) { cout << "list1: "; copy (l1.begin(), l1.end(), ostream_iterator<int>(cout," ")); cout << endl << "list2: "; copy (12.begin(), 12.end(), ostream_iterator<int>(cout," ")); cout << endl << endl; } int main() { list<int> list1, list2; for (int i=0; i<6; ++i) { list1.push_back(i); list2.push_front(i); } printLists(list1, list2); list2.splice(find(list2.begin(),list2.end(), 3), list1); printLists(list1, list2); list2.splice(list2.end(), list2, list2.begin()); printLists(list1, list2); list2.sort(); list1 = list2; list2.unique(); printLists(list1, list2); list1.merge(list2); printLists(list1, list2); } }}} = stack栈 = #include <stack> namespace std { template <class T, class Container = deque<T> > class stack; } 实现: 主要操作 push() 入栈 top() 取栈顶元素 pop() 出栈 例: {{{#!cplusplus int main() { stack<int> st; st.push(l); st.push(2); st.push(3); cout << st.top() << ' '; st.pop() ; cout << st.top() << ' '; st.pop() ; st.top() = 77; st.push(4); st.push(5); st.pop() ; while (!st.empty()) { cout << st.top() << ' '; st.pop() ; } cout << endl; } }}} = queue队列 = #include <queue> namespace std { template <class T, class Container = deque<T> > class queue; } 实现 主要操作: push pop back front 例 {{{#!cplusplus int main() { queue<string> q; q.push("These "); q.push("are "); q.push("more than "); cout << q.front(); q.pop(); cout << q.front(); q.pop(); q.push(''four "); q.push("words!"); q.pop(); cout << q.front(); q.pop(); cout << q.front() << endl; q.pop(); cout << "number of elements in the queue: " << q.size() << endl; } }}} = priority_queue优先队列 = 按照大小顺序出队的队列 #include <queue> namespace std { template <class T, class Container = vector<T>, class Compare = less<typename Container::value_type> > class priority_queue; } 实现:堆 push() 入队 top() 读取下一个元素 pop() 删除下一个元素 {{{#!cplusplus int main() { priority_queue<float> q; q.push(66.6); q.push(22.2); q.push(44.4); cout << q.top() << ' '; q.pop(); cout << q.top() << endl; q.pop(); q.push(11.1); q.push(55.5); q.push(33.3); q.pop(); while (!q.empty()) { cout << q.top() << ' '; q.pop(); } cout << endl; } }}} = set与multi_set = 在这两种容器中,元素能够根据指定的排序规则自动的排序,以优化查找。两者区别是:set不允许有重复的元素,multi_set允许有重复的元素。 {{{#!cplusplus #include <set> namespace std { template <class T, class Compare = less<T>, class Allocator = allocator<T> > class set; template <class T, class Compare = less<T>, class Allocator = allocator<T> > class multiset; } }}} 内部结构 例子: {{{#!cplusplus #include <iostream> #include <set> using namespace std; int main() { typedef set<int,greater<int> > IntSet; IntSet coll1; // empty set container coll1.insert(4); coll1.insert(3); coll1.insert(5); coll1.insert(1); coll1.insert(6); coll1.insert(2); coll1.insert(5); IntSet::iterator pos; for (pos = coll1.begin(); pos != coll1.end(); ++pos) { cout << *pos << ' '; } cout << endl; pair<IntSet::iterator,bool> status = coll1.insert(4); if (status.second) { cout << "4 inserted as element "<< distance (coll1.begin(),status. first) + 1<< endl; }else { cout << "4 already exists" << endl; } set<int> coll2(coll1.begin(), coll1.end()); copy (coll2.begin(), coll2.end(), ostream_iterator<int>(cout," ")); cout << endl; coll2.erase (coll2.begin(), coll2.find(3)); int num; num = coll2.erase (5); cout << num << " element(s) removed" << endl; copy (coll2.begin(), coll2.end(), ostream_iterator<int>(cout," ")); cout << endl; } }}} = map与multi_map = 存放关键字与对应值的数据结构 {{{#!cplusplus #include <map> namespace std { template <class Key, class T, class Compare = less<Key>, class Allocator = allocator<pair<const Key,T> > > class map; template <class Key, class T, class Compare = less<Key>, class Allocator = allocator<pair<const Key,T> > > class multimap; } }}} 内部结构 {{{#!cplusplus #include <iostream> #include <map> #include <string> using namespace std; int main() { typedef map<string,float> StringFloatMap; StringFloatMap stocks; // create empty container stocks["BASF"] = 369.50; stocks["VW"] = 413.50; stocks["Daimler"] = 819.00; stocks["BMW"] = 834.00; stocks["Siemens"] = 842.20; StringFloatMap::iterator pos; for (pos = stocks.begin(); pos != stocks.end(); ++pos) { cout << "stock: " << pos->first << "\t" << "price: " << pos->second << endl; } cout << endl; for (pos = stocks.begin(); pos != stocks.end(); ++pos) { pos->second *= 2; } for (pos = stocks.begin(); pos != stocks.end(); ++pos) { cout << "stock: " << pos->first << "\t"<< "price: " << pos->second << endl; } cout << endl; stocks["Volkswagen"] = stocks["VW"]; stocks.erase("VW"); for (pos = stocks.begin(); pos != stocks.end(); ++pos) { cout << "stock: " << pos->first << "\t"<< "price: " << pos->second << endl; } } }}} |
STL概述
STL是Standard Template Library的缩写,是C++标准库中最强大、最复杂、最有用的部分。STL主要由容器(container)、跌代子(iterator)、算法(algorithm)所组成。还有仿函数(functor)、适配器(adapter)、配置器(allocator)等辅助组件。
1. 容器container
容器是存放和管理数据元素的数据结构,分为两大类:顺序容器(sequence container)和关联容器(associative container)。
- 顺序容器有:vector(向量,酷似数组), deque(双端队列), list(双链表)
- 关联容器有:map(字典), set(集合), multi_map(允许重复键的字典), multi_set(允许重复键的集合)
除此以外还有:
- 特殊的容器:string(字符串), array(C语言原始数组)
- 容器适配器:stack(栈), queue(队列), priority_queue(优先队列)
- 内部容器:不提供给用户使用,只用来实现其他容器,比如红黑树(用来实现map,set),堆(用来实现priority_queue)
容器一般使用模板类来实现
2. 跌代子iterator
跌代子是用来访问容器内元素的对象,类似指针。 跌代子根据能力的不同,分为:
- 随机跌代子(vector、deque的迭代子)
- 双向跌代子(list的迭代子)
- 单向跌代子
- 输入跌代子
- 输出跌代子
另外还有
- 跌代子适配器:将原来不是迭代子的东西变成迭代子,或者将一种迭代子变成另一种迭代子(比如back_inserter, front_inserter, inserter, 反向迭代子,ostream_iterator, istream_iterator)
3. 算法algorithm
算法是用来处理容器内的元素的一些操作,比如搜索、排序、拷贝、修改等。算法一般使用函数模板来实现。
4. 仿函数functor
用法类似函数的对象。用重载了operator()的类或者模板类来实现
vector向量
1. 接口说明
vector的用法类似于数组,不同的是数组空间可以动态分配。
T可以是任何类型,但是必须满足:assignable, copyable
1.1. 构造方法
1.2. 不变操作和赋值
1.3. 元素访问
1.4. 跌代子
1.5. 插入删除操作
1 a1.insert( a1.begin(), 5); //在a1的最前面插入一个5
2 a1.insert(a1.end(), 10, 6); //在a1的最后面插入10个6
3 a1.insert(a1.begin(), values, values+5) //在a1的最前面插入values[0]到values[4]
4 a1.push_back( 5 ) //在a1的最后面插入一个5
5 a1.pop_back( ) // 删除a1的最后一个元素
6 a1.erase( a1.begin() ) //删除a1中的第一个元素
7 a1.erase( a1.begin(), a1.begin() +2) //删除a1最前面2个元素
8 a1.resize( 10 ) //将a1元素个数改为10,增加的部分值为默认构造
9 a1.resize( 10, 6) //将a1元素个数改为10,增加的部分值为6
10 a1.clear() //清除所有元素
11
2. 用法实例
1 #include <vector>
2 int main() {
3 using namespace std;
4 vector< string > sentence;
5 sentence.reserve( 5 );
6 sentence.push_back( “ Hello, “);
7 sentence.push_back( “how “);
8 sentence.push_back( “are “);
9 sentence.push_back( "you ");
10 sentence.push_back( “?“);
11 copy( sentence.begin(), sentence.end(), ostream_iterator<string>(cout, “ “));
12 cout << endl;
13 cout << sentence.size() << endl;
14 cout << sentence.capacity() << endl;
15 swap( sentence[1], sentence[3]);
16 sentence.insert( find(sentence.begin(), sentence.end(), “?”), “always”);
17 sentence.back() = “!”;
18 copy( sentence.rbegin(), sentence.rend(), ostream_iterator<string>(cout, “ “));
19 cout << endl;
20 }
用vector代替数组使用
3. vector的实现
vector的实现类似于数据结构中的顺序表
attachment:vector.jpg
1 template<class T, class Alloc=alloc>
2 class vector {
3 public:
4 typedef T value_type;
5 typedef value_type* iterator;
6 typedef value_type& reference;
7 typedef size_t size_type;
8 protected:
9 iterator start;
10 iterator finish;
11 iterator end_of_storage;
12
13
14 public:
15 iterator begin() { return start; }
16 iterator end() { return finish; }
17 size_type size() const { //返回当前元素个数
18 return size_type(end() - begin());
19 }
20 bool empty() const {
21 return begin() == end();
22 }
23 reference operator[](size_type n) {
24 return *(begin() + n);
25 }
26 reference front() {
27 return *begin();
28 }
29 reference back() {
30 return *(end() - 1);
31 }
32 size_type capacity() const { //返回当前容量的大小
33 return size_type(end_of_storage - begin());
34 }
35 size_type reserve(size_type n); //改变容量的大小
36 void push_back(const T& x) {
37 if(finish != end_of_storage) {
38 construct(finish, x);
39 ++finish;
40 } else {
41 insert_aux(end(), x);
42 }
43 }
44 protected:
45 typedef simple_alloc<value_type, Alloc> data_allocator;
46 void deallocate() {
47 if(start) data_allocator::deallocate(start, end_of_storage - start);
48 }
49
50 void insert_aux(iterator position, const T& x) {
51 if(finish != end_of_storage) {
52 construct(finish, *(finish-1));
53 ++finish;
54 T x_copy = x;
55 copy_backward(position, finish-2, finish-1);
56 *position = x_copy;
57 } else {
58 const size_type old_size = size();
59 const size_type len = old_size != 0 ? 2 * old_size : 1;
60 iterator new_start = data_allocator::alloate(len);
61 iterator new_finish = new_start;
62 try {
63 new_finish = uninitialized_copy(start, position, new_start);
64 construct(new_finish, x);
65 ++ new_finish;
66 new_finish = uninitialized_copy(position, finish, new_finish);
67 } catch(...) {
68 destroy(new_start, new_finish);
69 data_allocator::deallocate(new_start, len);
70 throw;
71 }
72 destroy(begin(), end());
73 deallocate();
74 start = new_start;
75 finish = new_finish;
76 end_of_storage = new_start + len;
77 }
78 }
79 };
deque双端队列
deque与vector相似,区别是deque两端都是开放的,两端插入删除都很快。在头文件<deque>中定义。 attachment:deque.jpg
实现: attachment:deque_imp.jpg
可以随机访问,但速度比vector稍慢
迭代子是随机跌代子,是一种class而不是原始指针
操作非常相似,增加操作:push_front, pop_front,减少操作:reserve,capacity
任何插入和删除操作都可能使跌代子失效
例子:
例子:
1 int main() {
2 deque<string> coll;
3 coll.assign (3, string("string"));
4 coll.push_back ("last string");
5 coll.push_front ("first string");
6 copy (coll.begin(), coll.end(), ostream_iterator<string>(cout,"\n"));
7 coll.pop_front();
8 coll.pop_back();
9 for (int i=1; i<coll.size(); ++i) {
10 coll[i] = "another " + coll [i];
11 }
12 coll.resize (4, "resized string");
13 copy (coll.begin(), coll.end(), ostream_iterator<string>(cout,"\n"));
14 }
list链表
一般为双链表实现
#include <list> 提供双向跌代子,不能随机访问 插入删除操作非常快速 插入删除操作不会使跌代子失效 提供了一些移动元素的算法,比通用算法更快
例
1 void printLists (const list<int>& 11, const list<int>& 12) {
2 cout << "list1: ";
3 copy (l1.begin(), l1.end(), ostream_iterator<int>(cout," "));
4 cout << endl << "list2: ";
5 copy (12.begin(), 12.end(), ostream_iterator<int>(cout," "));
6 cout << endl << endl;
7 }
8 int main() {
9 list<int> list1, list2;
10 for (int i=0; i<6; ++i) {
11 list1.push_back(i);
12 list2.push_front(i);
13 }
14 printLists(list1, list2);
15 list2.splice(find(list2.begin(),list2.end(), 3), list1);
16 printLists(list1, list2);
17 list2.splice(list2.end(), list2, list2.begin());
18 printLists(list1, list2);
19 list2.sort();
20 list1 = list2;
21 list2.unique();
22 printLists(list1, list2);
23 list1.merge(list2);
24 printLists(list1, list2);
25 }
stack栈
#include <stack> namespace std { template <class T, class Container = deque<T> > class stack; } 实现:
主要操作 push() 入栈 top() 取栈顶元素 pop() 出栈 例:
1 int main() {
2 stack<int> st;
3 st.push(l);
4 st.push(2);
5 st.push(3);
6 cout << st.top() << ' ';
7 st.pop() ;
8 cout << st.top() << ' ';
9 st.pop() ;
10 st.top() = 77;
11 st.push(4);
12 st.push(5);
13 st.pop() ;
14 while (!st.empty()) {
15 cout << st.top() << ' ';
16 st.pop() ;
17 }
18 cout << endl;
19 }
queue队列
#include <queue> namespace std { template <class T, class Container = deque<T> > class queue; } 实现
主要操作: push pop back front
例
1 int main() {
2 queue<string> q;
3 q.push("These ");
4 q.push("are ");
5 q.push("more than ");
6 cout << q.front();
7 q.pop();
8 cout << q.front();
9 q.pop();
10 q.push(''four ");
11 q.push("words!");
12 q.pop();
13 cout << q.front();
14 q.pop();
15 cout << q.front() << endl;
16 q.pop();
17 cout << "number of elements in the queue: " << q.size() << endl;
18 }
priority_queue优先队列
按照大小顺序出队的队列 #include <queue> namespace std { template <class T, class Container = vector<T>, class Compare = less<typename Container::value_type> > class priority_queue; } 实现:堆
push() 入队 top() 读取下一个元素 pop() 删除下一个元素
1 int main() {
2 priority_queue<float> q;
3 q.push(66.6);
4 q.push(22.2);
5 q.push(44.4);
6 cout << q.top() << ' ';
7 q.pop();
8 cout << q.top() << endl;
9 q.pop();
10 q.push(11.1);
11 q.push(55.5);
12 q.push(33.3);
13 q.pop();
14 while (!q.empty()) {
15 cout << q.top() << ' ';
16 q.pop();
17 }
18 cout << endl;
19 }
set与multi_set
在这两种容器中,元素能够根据指定的排序规则自动的排序,以优化查找。两者区别是:set不允许有重复的元素,multi_set允许有重复的元素。
内部结构
例子:
1 #include <iostream>
2 #include <set>
3 using namespace std;
4 int main() {
5 typedef set<int,greater<int> > IntSet;
6 IntSet coll1; // empty set container
7 coll1.insert(4);
8 coll1.insert(3);
9 coll1.insert(5);
10 coll1.insert(1);
11 coll1.insert(6);
12 coll1.insert(2);
13 coll1.insert(5);
14 IntSet::iterator pos;
15 for (pos = coll1.begin(); pos != coll1.end(); ++pos) {
16 cout << *pos << ' ';
17 }
18 cout << endl;
19 pair<IntSet::iterator,bool> status = coll1.insert(4);
20 if (status.second) {
21 cout << "4 inserted as element "<< distance (coll1.begin(),status. first) + 1<< endl;
22 }else {
23 cout << "4 already exists" << endl;
24 }
25 set<int> coll2(coll1.begin(),
26 coll1.end());
27 copy (coll2.begin(), coll2.end(), ostream_iterator<int>(cout," "));
28 cout << endl;
29 coll2.erase (coll2.begin(), coll2.find(3));
30 int num;
31 num = coll2.erase (5);
32 cout << num << " element(s) removed" << endl;
33 copy (coll2.begin(), coll2.end(), ostream_iterator<int>(cout," "));
34 cout << endl;
35 }
map与multi_map
存放关键字与对应值的数据结构
内部结构
1 #include <iostream>
2 #include <map>
3 #include <string>
4 using namespace std;
5 int main() {
6 typedef map<string,float> StringFloatMap;
7 StringFloatMap stocks; // create empty container
8 stocks["BASF"] = 369.50;
9 stocks["VW"] = 413.50;
10 stocks["Daimler"] = 819.00;
11 stocks["BMW"] = 834.00;
12 stocks["Siemens"] = 842.20;
13 StringFloatMap::iterator pos;
14 for (pos = stocks.begin(); pos != stocks.end(); ++pos) {
15 cout << "stock: " << pos->first << "\t" << "price: " << pos->second << endl;
16 }
17 cout << endl;
18 for (pos = stocks.begin(); pos != stocks.end(); ++pos) {
19 pos->second *= 2;
20 }
21 for (pos = stocks.begin(); pos != stocks.end(); ++pos) {
22 cout << "stock: " << pos->first << "\t"<< "price: " << pos->second << endl;
23 }
24 cout << endl;
25 stocks["Volkswagen"] = stocks["VW"];
26 stocks.erase("VW");
27 for (pos = stocks.begin(); pos != stocks.end(); ++pos) {
28 cout << "stock: " << pos->first << "\t"<< "price: " << pos->second << endl;
29 }
30 }