上一节主要学习C++中的函数模版、数据结构以及排序查找操作:c++:-6,本节学习C++的范型程序设计和STL:

范型程序设计

  • 编写不依赖于具体数据类型的程序
  • 将算法从特定的数据结构中抽象出来,成为通用的
  • C++的模板为泛型程序设计奠定了关键的基础

概念

(1)用来界定具备一定功能的数据类型。例如:

  • 将“可以比大小的所有数据类型(有比较运算符)”这一概念记为Comparable
  • 将“具有公有的复制构造函数并可以用‘=’赋值的数据类型”这一概念记为Assignable
  • 将“可以比大小、具有公有的复制构造函数并可以用‘=’赋值的所有数据类型”这个概念记作Sortable

(2)对于两个不同的概念A和B,如果概念A所需求的所有功能也是概念B所需求的功能,那么就说概念B是概念A的子概念。例如:

  • Sortable既是Comparable的子概念,也是Assignable的子概念

模型

符合一个概念的数据类型称为该概念的模型,例如:

  • int型是Comparable概念的模型。
  • 静态数组类型不是Assignable概念的模型(无法用“=”给整个静态数组赋值)

用概念做模板参数名

  • 很多STL的实现代码就是使用概念来命名模板参数的。
  • 为概念赋予一个名称,并使用该名称作为模板参数名。
  • 例如:表示insertionSort这样一个函数模板的原型:
  1. template <class Sortable>
  2. void insertionSort(Sortable a[], int n);

STL

标准模板库(Standard Template Library,简称STL)提供了一些非常常用的数据结构和算法

更多参考:C++:STL

  • 标准模板库(Standard Template Library,简称STL)定义了一套概念体系,为泛型程序设计提供了逻辑基础
  • STL中的各个类模板、函数模板的参数都是用这个体系中的概念来规定的。
  • 使用STL的模板时,类型参数既可以是C++标准库中已有的类型,也可以是自定义的类型——只要这些类型是所要求概念的模型。

介绍

STL的基本组件

  • 容器(container)
  • 迭代器(iterator)
  • 函数对象(function object)
  • 算法(algorithms)

STL的基本组件间的关系

(1)* Iterators(迭代器)是算法和容器的桥梁。

  • 将迭代器作为算法的参数、通过迭代器来访问容器而不是把容器直接作为算法的参数。

(2)将函数对象作为算法的参数而不是将函数所执行的运算作为算法的一部分。

(3)使用STL中提供的或自定义的迭代器和函数对象,配合STL的算法,可以组合出各种各样的功能。

容器(container)

  • 容器类是容纳、包含一组元素或元素集合的对象。
  • 基于容器中元素的组织方式:顺序容器、关联容器
  • 按照与容器所关联的迭代器类型划分:可逆容器随机访问容器

容器分类

  • unorderedset (无序集合)、unorderedmultiset(无序多重集合)
  • unorderedmap(无序映射)、unordermultimap(无序多重映射)
  • set(集合)、multiset(多重集合)、map(映射)、multimap(多重映射)
  • array(数组)、vector(向量)、deque(双端队列)、forward_list(单链表)、list(列表)
  • 顺序容器
  • (有序)关联容器
  • 无序关联容器

容器适配器

  • stack(栈)、queue(队列)、priority_queue(优先队列)

使用容器,需要包含对应的头文件

通用功能

  • 用默认构造函数构造空容器
  • 支持关系运算符:==、!=、<、<=、>、>=
  • begin()、end():获得容器首、尾迭代器
  • clear():将容器清空
  • empty():判断容器是否为空
  • size():得到容器元素个数
  • s1.swap(s2):将s1和s2两容器内容交换

对可逆容器的访问

(1)STL为每个可逆容器都提供了逆向迭代器,逆向迭代器可以通过下面的成员函数得到:

  • rbegin() :指向容器尾的逆向迭代器
  • rend():指向容器首的逆向迭代器

    (2)逆向迭代器的类型名的表示方式如下:
  • S::reverse_iterator:逆向迭代器类型
  • S::constreverseiterator:逆向常迭代器类型

随机访问容器

随机访问容器支持对容器的元素进行随机访问

s[n]:获得容器s的第n个元素

迭代器(iterator)

  • 迭代器是泛化的指针,提供了顺序访问容器中每个元素的方法
  • 提供了顺序访问容器中每个元素的方法;
  • 可以使用“++”运算符来获得指向下一个元素的迭代器;
  • 可以使用“*”运算符访问一个迭代器所指向的元素,如果元素类型是类或结构体,还可以使用“->”运算符直接访问该元素的一个成员;
  • 有些迭代器还支持通过“--”运算符获得指向上一个元素的迭代器;
  • 迭代器是泛化的指针:指针也具有同样的特性,因此指针本身就是一种迭代器;
  • 使用独立于STL容器的迭代器,需要包含头文件

函数对象(function object)

  • 一个行为类似函数的对象,对它可以像调用函数一样调用。
  • 函数对象是泛化的函数:任何普通的函数和任何重载了“()” 运算符的类的对象都可以作为函数对象使用
  • 使用STL的函数对象,需要包含头文件

算法(algorithms)

  • STL包括70多个算法

    例如:排序算法,消除算法,计数算法,比较算法,变换算法,置换算法和容器管理等
  • 可以广泛用于不同的对象和内置的数据类型。
  • 使用STL的算法,需要包含头文件

举例

从标准输入读入几个整数,存入向量容器,输出它们的相反数

  1. #include "iostream"
  2. #include "vector"
  3. #include "iterator"
  4. #include "algorithm"
  5. #include "functional"
  6. using namespace std;
  7. int main()
  8. {
  9. const int N=5;
  10. vector<int> s(N);
  11. for(int i=0;i<N;i++)
  12. {
  13. cin >> s[i];
  14. }
  15. transform(s.begin(),s.end(),ostream_iterator<int>(cout," "),negate<int>());
  16. cout <<endl;
  17. return 0;
  18. }

transform算法的一种实现:

  1. template <class InputIterator, class OutputIterator, class UnaryFunction>
  2. OutputIterator transform(InputIterator first, InputIterator last, OutputIterator result, UnaryFunction op) {
  3. for (;first != last; ++first, ++result)
  4. *result = op(*first);
  5. return result;
  6. }
  • transform算法顺序遍历first和last两个迭代器所指向的元素;
  • 将每个元素的值作为函数对象op的参数;
  • 将op的返回值通过迭代器result顺序输出;
  • 遍历完成后result迭代器指向的是输出的最后一个元素的下一个位置,transform会将该迭代器返回

迭代器

(1)迭代器是算法和容器的桥梁

  • 迭代器用于访问容器中的元素
  • 算法不直接操作容器中的数据,而是通过迭代器间接操作

(2)算法和容器独立

  • 增加新的算法,无需影响容器的实现
  • 增加新的容器,原有的算法也能适用

输入输出迭代器

输入流迭代器

  1. istream_iterator<T>
  • 以输入流(如cin)为参数构造
  • 可用*(p++)获得下一个输入的元素

输出流迭代器

  1. ostream_iterator<T>
  • 构造时需要提供输出流(如cout)
  • 可用(*p++) = x将x输出到输出流

二者都属于适配器

  • 适配器是用来为已有对象提供新的接口的对象
  • 输入流适配器和输出流适配器为流对象提供了迭代器的接口

分类

迭代器是泛化的指针,提供了类似指针的操作(诸如++、*、->运算符)

(1)输入迭代器

可以用来从序列中读取数据,如输入流迭代器

(2)输出迭代器

允许向序列中写入数据,如输出流迭代器

(3)前向迭代器

既是输入迭代器又是输出迭代器,并且可以对序列进行单向的遍历

(4)双向迭代器

与前向迭代器相似,但是在两个方向上都可以对数据遍历

(5)随机访问迭代器

也是双向迭代器,但能够在序列中的任意两个位置之间进行跳转,如指针、使用vector的begin()、end()函数得到的迭代

迭代器的区间

(1)两个迭代器表示一个区间:[p1, p2)

(2)STL算法常以迭代器的区间作为输入,传递输入数据

(3)合法的区间

p1经过n次(n > 0)自增(++)操作后满足p1 == p2

(4)区间包含p1,但不包含p2

举例

(1)从标准输入读入几个实数,分别将它们的平方输出

  1. #include <iterator>
  2. #include <iostream>
  3. #include <algorithm>
  4. using namespace std;
  5. //求平方的函数
  6. double square(double x) {
  7. return x * x;
  8. }
  9. int main() {
  10. //从标准输入读入若干个实数,分别将它们的平方输出
  11. transform(istream_iterator<double>(cin), istream_iterator<double>(),
  12. ostream_iterator<double>(cout, "\t"), square);
  13. cout << endl;
  14. return 0;
  15. }
  16. 输出:
  17. 1 2 3 4
  18. 1 4 9 16

(2) 综合运用几种迭代器的示例:程序涉及到输入迭代器、输出迭代器、随机访问迭代器这三个迭代器概念,并且以前两个概念为基础编写了一个通用算法。

  1. #include <algorithm>
  2. #include <iterator>
  3. #include <vector>
  4. #include <iostream>
  5. using namespace std;
  6. //模版迭代器
  7. //将来自输入迭代器的n个T类型的数值排序,将结果通过输出迭代器result输出
  8. template <class T, class InputIterator, class OutputIterator>
  9. void mySort(InputIterator first, InputIterator last, OutputIterator result) {
  10. //通过输入迭代器将输入数据存入向量容器s中
  11. vector<T> s;
  12. for (;first != last; ++first)
  13. {
  14. s.push_back(*first);//对s进行排序,sort函数的参数必须是随机访问迭代器
  15. }
  16. sort(s.begin(), s.end());
  17. copy(s.begin(), s.end(), result); //将s序列通过输出迭代器输出
  18. }
  19. int main() {
  20. //将s数组的内容排序后输出
  21. double a[5] = { 1.2, 2.4, 0.8, 3.3, 3.2 };
  22. mySort<double>(a, a + 5, ostream_iterator<double>(cout, " "));
  23. cout << endl;
  24. //从标准输入读入若干个整数,将排序后的结果输出
  25. mySort<int>(istream_iterator<int>(cin), istream_iterator<int>(), ostream_iterator<int>(cout, " "));
  26. cout << endl;
  27. return 0;
  28. }

上面代码有些问题。

顺序容器

包括:

  • 向量(vector)
  • 双端队列(deque)
  • 列表(list)
  • 单向链表(forward_list)

以上四种在逻辑上可看作是一个长度可扩展的数组

基本功能

(1)特点

  • 元素线性排列,可以随时在指定位置插入元素和删除元素。
  • 必须符合Assignable这一概念(即具有公有的拷贝构造函数并可以用“=”赋值)。
  • array对象的大小固定,forward_list有特殊的添加和删除操作。

(2)接口(不包含单向链表(forward_list)和数组(array))

  • 构造函数
  • 赋值函数:assign
  • 插入函数:insert, pushfront(只对list和deque), pushback,emplace,emplace_front
  • 删除函数:erase,clear,popfront(只对list和deque) ,popback,emplace_back
  • 首尾元素的直接访问:front,back
  • 改变大小:resize

(3)举例

  1. #include <iostream>
  2. #include <list>
  3. #include <deque>
  4. using namespace std;
  5. //输出指定的顺序容器的元素
  6. template <class T>
  7. void printContainer(const char* msg, const T& s) {
  8. cout << msg << ": ";
  9. copy(s.begin(), s.end(), ostream_iterator<int>(cout, " "));//s.begin()指向s的第一个元素(头),s.end()指向s的最后一个元素(尾)的下一个
  10. cout << endl;
  11. }
  12. int main() {
  13. //从标准输入读入10个整数,将它们分别从s的头部加入(头)
  14. deque<int> s;
  15. for (int i = 0; i < 10; i++) {
  16. int x;
  17. cin >> x;
  18. s.push_front(x);
  19. }
  20. printContainer("deque at first", s);
  21. //用s容器的内容的逆序构造列表容器l
  22. list<int> l(s.rbegin(), s.rend());//rbegin()指向最后一个元素(尾),rend()指向第一个元素(头)
  23. printContainer("list at first", l);
  24. //将列表容器l的每相邻两个元素顺序颠倒
  25. /*
  26. * 例如:1 2 3 4
  27. * 先取v=1,然后删除1,即:2 3 4
  28. * 然后将1插入到2后面,即:2 1 3 4
  29. */
  30. list<int>::iterator iter = l.begin();//iter是双向链表
  31. while (iter != l.end()) {
  32. int v = *iter;
  33. iter = l.erase(iter);//l.erase(iter)删除该位置,返回下一个位置
  34. l.insert(++iter, v);//插入到最后
  35. }
  36. printContainer("list at last", l);
  37. //用列表容器l的内容给s赋值,将s输出
  38. s.assign(l.begin(), l.end());
  39. printContainer("deque at last", s);
  40. return 0;
  41. }
  42. 输出:
  43. 1 2 3 4 5 6 7 8 9 10
  44. deque at first: 10 9 8 7 6 5 4 3 2 1
  45. list at first: 1 2 3 4 5 6 7 8 9 10
  46. list at last: 2 1 4 3 6 5 8 7 10 9
  47. deque at last: 2 1 4 3 6 5 8 7 10 9

队列:



列表:

特性

顺序容器:向量、双端队列、列表、单向链表、数组

向量(Vector)

  • 容量(capacity):实际分配空间的大小
  • s.capacity() :返回当前容量
  • s.reserve(n):若容量小于n,则对s进行扩展,使其容量至少为n
  • 一个可以扩展的动态数组
  • 随机访问、在尾部插入或删除元素快
  • 在中间或头部插入或删除元素慢

双端队列(deque)

  • 在两端插入或删除元素快
  • 在中间插入或删除元素慢
  • 随机访问较快,但比向量容器慢

举例:先按照从大到小顺序输出奇数,再按照从小到大顺序输出偶数。

  1. #include "iostream"
  2. #include "vector"
  3. #include "deque"
  4. using namespace std;
  5. int main() {
  6. istream_iterator<int> i1(cin), i2; //建立一对输入流迭代器
  7. vector<int> s1(i1, i2); //通过输入流迭代器从标准输入流中输入数据
  8. sort(s1.begin(), s1.end()); //将输入的整数排序
  9. deque<int> s2;
  10. //以下循环遍历s1
  11. for (vector<int>::iterator iter = s1.begin(); iter != s1.end(); ++iter)
  12. {
  13. if (*iter % 2 == 0) //偶数放到s2尾部
  14. s2.push_back(*iter);
  15. else //奇数放到s2首部
  16. s2.push_front(*iter);
  17. }
  18. //将s2的结果输出
  19. copy(s2.begin(), s2.end(), ostream_iterator<int>(cout, " "));
  20. cout << endl;
  21. return 0;
  22. }

列表(list)

在任意位置插入和删除元素都很快

不支持随机访问

接合(splice)操作

  • s1.splice(p, s2, q1, q2):将s2中[q1, q2)移动到s1中p所指向元素之前

举例:list结合

  1. #include "iostream"
  2. #include "string"
  3. #include "list"
  4. using namespace std;
  5. int main() {
  6. string names1[] = { "Alice", "Helen", "Lucy", "Susan" };
  7. string names2[] = { "Bob", "David", "Levin", "Mike" };
  8. //用names1数组的内容构造列表s1
  9. list<string> s1(names1, names1 + 4);
  10. //用names2数组的内容构造列表s2
  11. list<string> s2(names2, names2 + 4);
  12. //将s1的第一个元素放到s2的最后
  13. s2.splice(s2.end(), s1, s1.begin());
  14. list<string>::iterator iter1 = s1.begin(); //iter1指向s1首
  15. advance(iter1, 2); //iter1前进2个元素,它将指向s1第3个元素
  16. list<string>::iterator iter2 = s2.begin(); //iter2指向s2首
  17. ++iter2; //iter2前进1个元素,它将指向s2第2个元素
  18. list<string>::iterator iter3 = iter2; //用iter2初始化iter3
  19. advance(iter3, 2); //iter3前进2个元素,它将指向s2第4个元素
  20. //将[iter2, iter3)范围内的结点接到s1中iter1指向的结点前
  21. s1.splice(iter1, s2, iter2, iter3);
  22. //分别将s1和s2输出
  23. copy(s1.begin(), s1.end(), ostream_iterator<string>(cout, " "));
  24. cout << endl;
  25. copy(s2.begin(), s2.end(), ostream_iterator<string>(cout, " "));
  26. cout << endl;
  27. return 0;
  28. }
  29. 输出:
  30. Helen Lucy David Levin Susan
  31. Bob Mike Alice

单向链表(forward_list)

  • 单向链表每个结点只有指向下个结点的指针,没有简单的方法来获取一个结点的前驱;
  • 未定义insert、emplace和erase操作,而定义了insertafter、emplaceafter和erase_after操作,其参数与list的insert、emplace和erase相同,但并不是插入或删除迭代器p1所指的元素,而是对p1所指元素之后的结点进行操作;
  • 不支持size操作。

数组(array)

  • array是对内置数组的封装,提供了更安全,更方便的使用数组的方式
  • array的对象的大小是固定的,定义时除了需要指定元素类型,还需要指定容器大小。
  • 不能动态地改变容器大小

顺序容器的比较

  • STL所提供的顺序容器各有所长也各有所短,我们在编写程序时应当根据我们对容器所需要执行的操作来决定选择哪一种容器。
  • 如果需要执行大量的随机访问操作,而且当扩展容器时只需要向容器尾部加入新的元素,就应当选择向量容器vector;
  • 如果需要少量的随机访问操作,需要在容器两端插入或删除元素,则应当选择双端队列容器deque;
  • 如果不需要对容器进行随机访问,但是需要在中间位置插入或者删除元素,就应当选择列表容器list或forward_list;
  • 如果需要数组,array相对于内置数组类型而言,是一种更安全、更容易使用的数组类型。

插入迭代器和适配器

插入迭代器

  • 用于向容器头部、尾部或中间指定位置插入元素的迭代器
  • 包括前插迭代器(frontinserter)、后插迭代器(backinsrter)和任意位置插入迭代器(inserter)

    例如:
  1. list<int> s;
  2. back_inserter iter(s);
  3. *(iter++) = 5; //通过iter把5插入s末尾

适配器

以顺序容器为基础构建一些常用数据结构,是对顺序容器的封装

  • 栈(stack):最先压入的元素最后被弹出
  • 队列(queue):最先压入的元素最先被出
  • 优先级队列(priority_queue):最“大”的元素最先被弹出

栈和队列模板

(1)栈模板

  1. template <class T, class Sequence = deque<T> > class stack;

(2)队列模板

  1. template <class T, class FrontInsertionSequence = deque<T> > class queue;

栈可以用任何一种顺序容器作为基础容器,而队列只允许用前插顺序容器(双端队列或列表)

栈和队列共同支持的操作

  • s1 op s2 op可以是==、!=、<、<=、>、>=之一,它会对两个容器适配器之间的元素按字典序进行比较
  • s.size() 返回s的元素个数
  • s.empty() 返回s是否为空
  • s.push(t) 将元素t压入到s中
  • s.pop() 将一个元素从s中弹出,对于栈来说,每次弹出的是最后被压入的元素,而对于队列,每次被弹出的是最先被压入的元素
  • 不支持迭代器,因为它们不允许对任意元素进行访问

栈和队列不同的操作

(1)栈的操作

  • s.top() 返回栈顶元素的引用

(2)队列操作

  • s.front() 获得队头元素的引用
  • s.back() 获得队尾元素的引用

(3)举例:利用栈反向输出单词

  1. #include "iostream"
  2. #include "string"
  3. #include "stack"
  4. using namespace std;
  5. int main() {
  6. stack<char> s;
  7. string str;
  8. cin >> str; //从键盘输入一个字符串
  9. //将字符串的每个元素顺序压入栈中
  10. for (string::iterator iter = str.begin(); iter != str.end(); ++iter)
  11. s.push(*iter);
  12. //将栈中的元素顺序弹出并输出
  13. while (!s.empty()) {
  14. cout << s.top();//读头元素
  15. s.pop();//取头元素
  16. }
  17. cout << endl;
  18. return 0;
  19. }
  20. 输出:
  21. cmake-build-debug/untitled
  22. deltitnu/gubed-dliub-ekamc

优先级队列

  • 优先级队列也像栈和队列一样支持元素的压入和弹出,但元素弹出的顺序与元素的大小有关,每次弹出的总是容器中最“大”的一个元素。
  1. template <class T, class Sequence = vector<T> > class priority_queue;
  • 优先级队列的基础容器必须是支持随机访问的顺序容器。
  • 支持栈和队列的size、empty、push、pop几个成员函数,用法与栈和队列相同。
  • 优先级队列并不支持比较操作。
  • 与栈类似,优先级队列提供一个top函数,可以获得下一个即将被弹出元素(即最“大”的元素)的引用。

关联容器

特点:

每个关联容器都有一个键(key)

可以根据键高效地查找元素

分类



(1)单重关联容器(set和map)

键值是唯一的,一个键值只能对应一个元素

(2)多重关联容器(multiset和multimap)

键值是不唯一的,一个键值可以对应多个元素

(3)简单关联容器(set和multiset)

  • 容器只有一个类型参数,如set、multiset,表示键类型
  • 容器的元素就是键本身

(4)二元关联容器(map和multimap)

  • 容器有两个类型参数,如map、multimap,分别表示键和附加数据的类型
  • 容器的元素类型是pair,即由键类型和元素类型复合而成的二元组

(5)无序关联容器

  • C++11新标准中定义了4个无序关联容器:unorderedset、unorderedmap、unorderedmultiset、unorderedmultimap
  • 不是使用比较运算符来组织元素的,而是通过一个哈希函数和键类型的==运算符。
  • 提供了与有序容器相同的操作
  • 可以直接定义关键字是内置类型的无序容器。
  • 不能直接定义关键字类型为自定义类的无序容器,如果需要,必须提供我们自己的hash模板。

接口

  • 插入:insert
  • 删除:erase
  • 查找:find
  • 定界:lowerbound、upperbound、equal_range
  • 计数:count

集合(set)

集合用来存储一组无重复的元素。由于集合的元素本身是有序的,可以高效地查找指定元素,也可以方便地得到指定大小范围的元素在容器中所处的区间。

举例:输入一串实数,将重复的去掉,取最大和最小者的中值,分别输出小于等于此中值和大于等于此中值的实数

  1. #include <set>
  2. #include <iterator>
  3. #include <utility>
  4. #include <iostream>
  5. using namespace std;
  6. int main() {
  7. set<double> s;
  8. while (true) {
  9. double v;
  10. cin >> v;
  11. if (v == 0) break; //输入0表示结束
  12. //尝试将v插入
  13. pair<set<double>::iterator,bool> r=s.insert(v);
  14. if (!r.second) //如果v已存在,输出提示信息
  15. cout << v << " is duplicated" << endl;
  16. }
  17. //得到第一个元素的迭代器
  18. set<double>::iterator iter1=s.begin();
  19. //得到末尾的迭代器
  20. set<double>::iterator iter2=s.end();
  21. //得到最小和最大元素的中值
  22. double medium=(*iter1 + *(--iter2)) / 2;
  23. //输出小于或等于中值的元素
  24. cout<< "<= medium: ";
  25. copy(s.begin(), s.upper_bound(medium), ostream_iterator<double>(cout, " "));
  26. cout << endl;
  27. //输出大于或等于中值的元素
  28. cout << ">= medium: ";
  29. copy(s.lower_bound(medium), s.end(), ostream_iterator<double>(cout, " "));
  30. cout << endl;
  31. return 0;
  32. }
  33. 输出:
  34. 1 2.5 5 3.5 5 7 9 2.5 0
  35. 5 is duplicated
  36. 2.5 is duplicated
  37. <= medium: 1 2.5 3.5 5
  38. >= medium: 5 7 9

映射(MAP)

  • 映射与集合同属于单重关联容器,它们的主要区别在于,集合的元素类型是键本身,而映射的元素类型是由键和附加数据所构成的二元组。
  • 在集合中按照键查找一个元素时,一般只是用来确定这个元素是否存在,而在映射中按照键查找一个元素时,除了能确定它的存在性外,还可以得到相应的附加数据。

举例1:有五门课程,每门都有相应学分,从中选择三门,输出学分总和

  1. #include <iostream>
  2. #include <map>
  3. #include <utility>
  4. using namespace std;
  5. int main() {
  6. map<string, int> courses;
  7. //将课程信息插入courses映射中
  8. courses.insert(make_pair("CSAPP", 3));
  9. courses.insert(make_pair("C++", 2));
  10. courses.insert(make_pair("CSARCH", 4));
  11. courses.insert(make_pair("COMPILER", 4));
  12. courses.insert(make_pair("OS", 5));
  13. int n = 3; //剩下的可选次数
  14. int sum = 0; //学分总和
  15. while (n > 0) {
  16. string name;
  17. cin >> name; //输入课程名称
  18. map<string, int>::iterator iter = courses.find(name);//查找课程
  19. if (iter == courses.end()) { //判断是否找到
  20. cout << name << " is not available" << endl;
  21. } else {
  22. sum += iter->second; //累加学分
  23. courses.erase(iter); //将刚选过的课程从映射中删除
  24. n--;
  25. }
  26. }
  27. cout << "Total credit: " << sum << endl; //输出总学分
  28. return 0;
  29. }
  30. 输出:
  31. OS
  32. COMPILER
  33. C++
  34. Total credit: 11

举例2:统计一句话中每个字母出现的次数

  1. #include <iostream>
  2. #include <map>
  3. #include <cctype>
  4. using namespace std;
  5. int main() {
  6. map<char, int> s; //用来存储字母出现次数的映射
  7. char c; //存储输入字符
  8. do {
  9. cin >> c; //输入下一个字符
  10. if (isalpha(c)){ //判断是否是字母:isalpha(c)若为英文字母,返回非0(小写字母为2,大写字母为1)。若不是字母,返回0。
  11. c = tolower(c); //将字母转换为小写
  12. s[c]++; //将该字母的出现频率加1
  13. }
  14. } while (c != '.'); //碰到“.”则结束输入
  15. //输出每个字母出现次数
  16. for (map<char, int>::iterator iter = s.begin(); iter != s.end(); ++iter)
  17. cout << iter->first << ": " << iter->second << " ";
  18. cout << endl;
  19. return 0;
  20. }
  21. 输出:
  22. isalpha112a.
  23. a: 3 h: 1 i: 1 l: 1 p: 1 s: 1

多重集合和多重映射

函数对象

函数对象

  • 一个行为类似函数的对象
  • 可以没有参数,也可以带有若干参数
  • 其功能是获取一个值,或者改变操作的状态。

    例:

    (1)普通函数就是函数对象

    (2)重载了“()”运算符的类的实例是函数对象

比如使用两种方式定义表示乘法的函数对象

用到的算法:

  1. template<class InputIterator, class Type, class BinaryFunction>
  2. Type accumulate(InputIterator first, InputIterator last, Type val, BinaryFunction binaryOp);

对[first, last)区间内的数据进行累“加”,binaryOp为用二元函数对象表示的“加”运算符,val为累“加”的初值

(1)通过定义普通函数

  1. #include <iostream>
  2. #include <numeric> //包含数值算法头文件
  3. using namespace std;
  4. //定义一个普通函数
  5. int mult(int x, int y) { return x * y; };
  6. int main() {
  7. int a[] = { 1, 2, 3, 4, 5 };
  8. const int N = sizeof(a) / sizeof(int);
  9. cout << "The result by multipling all elements in a is "
  10. << accumulate(a, a + N, 1, mult) //对a内的数据进行"运算",binaryOp为用二元函数对象表示的某种"运算",val为计算的初值
  11. << endl;
  12. return 0;
  13. }
  14. 输出:The result by multipling all elements in a is 120

(2)通过重载类的“()”运算符(例10-14)

  1. #include <iostream>
  2. #include <numeric> //包含数值算法头文件
  3. using namespace std;
  4. class MultClass{ //定义MultClass类
  5. public:
  6. //重载操作符operator()
  7. int operator() (int x, int y) const { return x * y; }
  8. };
  9. int main() {
  10. int a[] = { 1, 2, 3, 4, 5 };
  11. const int N = sizeof(a) / sizeof(int);
  12. cout << "The result by multipling all elements in a is "
  13. << accumulate(a, a + N, 1, MultClass()) //将类multclass传递给通用算法
  14. << endl;
  15. return 0;
  16. }
  17. 输出:The result by multipling all elements in a is 120

STL提供的函数对象

(1)用于算术运算的函数对象:

一元函数对象(一个参数) :negate

二元函数对象(两个参数) :plus、minus、multiplies、divides、modulus

(2)用于关系运算、逻辑运算的函数对象(要求返回值为bool)

一元谓词(一个参数):logical_not

二元谓词(两个参数):equalto、notequalto、greater、less、greaterequal、lessequal、logicaland、logical_or

举例:利用STL标准函数对象

  1. #include <iostream>
  2. #include <numeric> //包含数值算法头文件
  3. #include <functional> //包含标准函数对象头文件
  4. using namespace std;
  5. int main() {
  6. int a[] = { 1, 2, 3, 4, 5 };
  7. const int N = sizeof(a) / sizeof(int);
  8. cout << "The result by multipling all elements in A is "
  9. << accumulate(a, a + N, 1, multiplies<int>())//multiplies<int>():二元函数对象(两个参数)
  10. << endl; //将标准函数对象传递给通用算法
  11. return 0;
  12. }

举例:利用STL中的二元谓词函数对象

  1. #include <functional>
  2. #include<iostream>
  3. #include<vector>
  4. #include<algorithm>
  5. using namespace std;
  6. int main() {
  7. int intArr[] = { 30, 90, 10, 40, 70, 50, 20, 80 };
  8. const int N = sizeof(intArr) / sizeof(int);
  9. vector<int> a(intArr, intArr + N);
  10. cout << "before sorting:" << endl;
  11. copy(a.begin(),a.end(),ostream_iterator<int>(cout,"\t"));//输出
  12. cout << endl;
  13. sort(a.begin(), a.end(), greater<int>());//重排序:greater<int>()逆序
  14. cout << "after sorting:" << endl;
  15. copy(a.begin(),a.end(),ostream_iterator<int>(cout,"\t"));//输出
  16. cout << endl;
  17. sort(a.begin(), a.end(), less<int>());//重排序:greater<int>()逆序
  18. cout << "again sorting:" << endl;
  19. copy(a.begin(),a.end(),ostream_iterator<int>(cout,"\t"));//输出
  20. cout << endl;
  21. return 0;
  22. }
  23. 输出:
  24. before sorting:
  25. 30 90 10 40 70 50 20 80
  26. after sorting:
  27. 90 80 70 50 40 30 20 10
  28. again sorting:
  29. 10 20 30 40 50 70 80 90

函数适配器

(1)绑定适配器:bind1st、bind2nd

  • 将n元函数对象的指定参数绑定为一个常数,得到n-1元函数对象

(2)组合适配器:not1、not2

  • 将指定谓词的结果取反

(3)函数指针适配器:ptr_fun

  • 将一般函数指针转换为函数对象,使之能够作为其它函数适配器的输入。
  • 在进行参数绑定或其他转换的时候,通常需要函数对象的类型信息,例如bind1st和bind2nd要求函数对象必须继承于binary_function类型。但如果传入的是函数指针形式的函数对象,则无法获得函数对象的类型信息。

(4)成员函数适配器:ptrfun、ptrfun_ref

  • 对成员函数指针使用,把n元成员函数适配为n + 1元函数对象,该函数对象的第一个参数为调用该成员函数时的目的对象
  • 也就是需要将“object->method()”转为“method(object)”形式。将“object->method(arg1)”转为二元函数“method(object, arg1)”。

绑定适配器

  • binder2nd的实例构造通常比较冗长,bind2nd函数用于辅助构造binder2nd,产生它的一个实例。
  • binder1st和bind1st,将一个具体值绑定到二元函数的第一个参数。

举例:找到数组中第一个大于40的元素

算法介绍:

  1. template<class InputIterator, class UnaryPredicate>
  2. InputIterator find_if(InputIterator first, InputIterator last, UnaryPredicate pred);

它的功能是查找数组[first, last)区间中第一个pred(x)为真的元素。

  1. #include <functional>
  2. #include<iostream>
  3. #include<vector>
  4. #include<algorithm>
  5. using namespace std;
  6. int main() {
  7. int intArr[] = { 30, 90, 10, 40, 70, 50, 20, 80 };
  8. const int N = sizeof(intArr) / sizeof(int);
  9. vector<int> a(intArr, intArr + N);
  10. vector<int>::iterator p = find_if(a.begin(), a.end(), bind2nd(greater<int>(), 40));
  11. if (p == a.end())
  12. cout << "no element greater than 40" << endl;
  13. else
  14. cout << "first element greater than 40 is: " << *p << endl;
  15. return 0;
  16. }

组合适配器

  • 对于一般的逻辑运算,有时可能还需要对结果求一次逻辑反。
  • unarynegate和binarynegate实现了这一适配功能。STL还提供了not1和not2辅助生成相应的函数对象实例,分别用于一元谓词和二元谓词的逻辑取反。

举例:ptr_fun、not1和not2产生函数适配器实例

  1. #include <functional>
  2. #include<iostream>
  3. #include<vector>
  4. #include<algorithm>
  5. using namespace std;
  6. bool g(int x, int y) {
  7. return x > y;
  8. }
  9. int main() {
  10. int intArr[] = { 30, 90, 10, 40, 70, 50, 20, 80 };
  11. const int N = sizeof(intArr) / sizeof(int);
  12. vector<int> a(intArr, intArr + N);
  13. vector<int>::iterator p;
  14. p = find_if(a.begin(), a.end(), bind2nd(ptr_fun(g), 40));//找到第一个大于40的数:90
  15. if (p == a.end())
  16. cout << "no element greater than 40" << endl;
  17. else
  18. cout << "first element greater than 40 is: " << *p << endl;
  19. p = find_if(a.begin(), a.end(), not1(bind2nd(greater<int>(), 15)));//not1逻辑(greater)取反,即比12小的
  20. if (p == a.end())
  21. cout << "no element is not greater than 15" << endl;
  22. else
  23. cout << "first element that is not greater than 15 is: " << *p << endl;
  24. p = find_if(a.begin(), a.end(), bind2nd(not2(less<int>()), 15));
  25. if (p == a.end())
  26. cout << "no element is greater than 15" << endl;
  27. else
  28. cout << "first element that is greater than 15 is: " << *p << endl;
  29. return 0;
  30. }
  31. 输出:
  32. first element greater than 40 is: 90
  33. first element that is not greater than 15 is: 10
  34. first element that is greater than 15 is: 30

举例:成员函数适配器实例

  1. #include <functional>
  2. #include <iostream>
  3. #include <vector>
  4. #include <algorithm>
  5. using namespace std;
  6. struct Car {
  7. int id;
  8. Car(int id) { this->id = id; }
  9. void display() const { cout << "car " << id << endl; }
  10. };
  11. int main() {
  12. vector<Car *> pcars;
  13. vector<Car> cars;
  14. for (int i = 0; i < 5; i++)
  15. pcars.push_back(new Car(i));
  16. for (int i = 5; i < 10; i++)
  17. cars.push_back(Car(i));
  18. cout << "elements in pcars: " << endl;
  19. for_each(pcars.begin(), pcars.end(), std::mem_fun(&Car::display));//将成员函数适配(mem_fun)为普通函数对象
  20. cout << endl;
  21. cout << "elements in cars: " << endl;
  22. for_each(cars.begin(), cars.end(), std::mem_fun_ref(&Car::display));//mem_fun_ref的参数是对象的引用;mem_fun的参数是对象的指针
  23. cout << endl;
  24. for (size_t i = 0; i < pcars.size(); ++i)
  25. delete pcars[i];
  26. return 0;
  27. }
  28. 输出:
  29. elements in pcars:
  30. car 0
  31. car 1
  32. car 2
  33. car 3
  34. car 4
  35. elements in cars:
  36. car 5
  37. car 6
  38. car 7
  39. car 8
  40. car 9

算法

算法特点

  • STL算法本身是一种函数模版
  • 通过迭代器获得输入数据
  • 通过函数对象对数据进行处理
  • 通过迭代器将结果输出
  • STL算法是通用的,独立于具体的数据类型、容器类型

算法分类

不可变序列算法

  • 不直接修改所操作的容器内容的算法
  • 用于查找指定元素、比较两个序列是否相等、对元素进行计数等

例如:

  1. template<class InputIterator, class UnaryPredicate>
  2. InputIterator find_if(InputIterator first, InputIterator last, UnaryPredicate pred);

查找[first, last)区间内pred(x)为真的首个元素

可变序列算法

  • 可以修改它们所操作的容器对象
  • 包括对序列进行复制、删除、替换、倒序、旋转、交换、分割、去重、填充、洗牌的算法及生成一个序列的算法

例:

  1. template<class ForwardIterator, class T>
  2. InputIterator find_if(ForwardIterator first, ForwardIterator last, const T& x);

将[first, last)区间内的元素全部改写为x。

排序和搜索算法

  • 对序列进行排序
  • 对两有序序列进行合并
  • 对有序序列进行搜索
  • 有序序列的集合操作
  • 堆算法

例如:

  1. template <class RandomAccessIterator , class UnaryPredicate>
  2. void sort(RandomAccessIterator first, RandomAccessIterator last, UnaryPredicate comp);

以函数对象comp为“<”,对 [first, last)区间内的数据进行排序

数值算法

  • 求序列中元素的“和”、部分“和”、相邻元素的“差”或两序列的内积
  • 求“和”的“+”、求“差”的“-”以及求内积的“+”和“·”都可由函数对象指定

例:

  1. template<class InputIterator, class OutputIterator, class BinaryFunction>
  2. OutputIterator partial_sum(InputIterator first, InputIterator last, OutputIterator result, BinaryFunction op);

对[first, last)内的元素求部分“和”(所谓部分“和”,是一个长度与输入序列相同的序列,其第n项为输入序列前n个元素的“和”),以函数对象op为“+”运算符,结果通过result输出,返回的迭代器指向输出序列最后一个元素的下一个元素

习题

STL

(1)定义 const int b[3] ={9,2,3};调用insertionSort函数, insertionSort(b,3),可以执行成功。

正确。

分析:const类型不是 Sortable概念的模型

(2)set::insert(const value_type&val); 函数的返回值类型是什么?

  • iterator
  • 指针
  • bool
  • pair<iterator,bool>(对)

    (3)

  1. #include <deque>
  2. #include <iostream>
  3. using namespace std;
  4. /*
  5. * 队列:3 1 4 8 6
  6. * (出)头 尾(进)
  7. */
  8. int main()
  9. {
  10. deque<int> iDeque;
  11. iDeque.push_back(3);//队列尾部插入元素:push_back()
  12. iDeque.push_back(1);
  13. iDeque.push_back(4);
  14. iDeque.push_back(8);
  15. iDeque.push_back(6);
  16. while (!iDeque.empty()) {
  17. cout << "取出元素" << iDeque.front() << endl;//队列头部查看第一个元素:front()
  18. iDeque.pop_front();//队列头部弹出第一个元素:pop_front()
  19. }
  20. return 0;
  21. }
  22. 输出:
  23. 取出元素3
  24. 取出元素1
  25. 取出元素4
  26. 取出元素8
  27. 取出元素6

(4)

  1. #include <iostream>
  2. #include <algorithm> //程序中使用了算法find和sort
  3. #include <functional> //程序中使用了函数对象greater
  4. using namespace std;
  5. int main()
  6. {
  7. int a[8] = {3, 2, 5, 6, 4, 1, 9, 8};
  8. cout << "元素9的索引是:" << find(a, a+8, 9) - a << endl;//find返回的是地址
  9. cout << "升序排序:" << endl;
  10. sort(a, a+8);//默认是升序
  11. for (int i = 0; i < 8; i ++)
  12. cout << a[i] << ' ';
  13. cout << endl;
  14. cout << "升序排序:" << endl;
  15. sort(a, a+8, less<int>());//使用函数对象实现降序排序
  16. for (int i = 0; i < 8; i ++)
  17. cout << a[i] << ' ';
  18. cout << endl;
  19. cout << "降序排序:" << endl;
  20. sort(a, a+8, greater<int>());//使用函数对象实现降序排序
  21. for (int i = 0; i < 8; i ++)
  22. cout << a[i] << ' ';
  23. cout << endl;
  24. return 0;
  25. }
  26. 输出:
  27. 升序排序:
  28. 1 2 3 4 5 6 8 9
  29. 升序排序:
  30. 1 2 3 4 5 6 8 9
  31. 降序排序:
  32. 9 8 6 5 4 3 2 1

c++:-7的更多相关文章

  1. java web 开发三剑客 -------电子书

    Internet,人们通常称为因特网,是当今世界上覆盖面最大和应用最广泛的网络.根据英语构词法,Internet是Inter + net,Inter-作为前缀在英语中表示“在一起,交互”,由此可知In ...

  2. 所有selenium相关的库

    通过爬虫 获取 官方文档库 如果想获取 相应的库 修改对应配置即可 代码如下 from urllib.parse import urljoin import requests from lxml im ...

  3. In-Memory:内存数据库

    在逝去的2016后半年,由于项目需要支持数据的快速更新和多用户的高并发负载,我试水SQL Server 2016的In-Memory OLTP,创建内存数据库实现项目的负载需求,现在项目接近尾声,系统 ...

  4. 从直播编程到直播教育:LiveEdu.tv开启多元化的在线学习直播时代

    2015年9月,一个叫Livecoding.tv的网站在互联网上引起了编程界的注意.缘于Pingwest品玩的一位编辑在上网时无意中发现了这个网站,并写了一篇文章<一个比直播睡觉更奇怪的网站:直 ...

  5. 【.net 深呼吸】细说CodeDom(8):分支与循环

    有人会问,为啥 CodeDom 不会生成 switch 语句,为啥没生成 while 语句之类.要注意,CodeDom只关心代码逻辑,而不是语法,语法是给写代码的人用的.如果用.net的“反编译”工具 ...

  6. 【原】Android热更新开源项目Tinker源码解析系列之三:so热更新

    本系列将从以下三个方面对Tinker进行源码解析: Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Android热更新开源项目Tinker源码解析系列之二:资源文件热更新 A ...

  7. App开发:模拟服务器数据接口 - MockApi

    为了方便app开发过程中,不受服务器接口的限制,便于客户端功能的快速测试,可以在客户端实现一个模拟服务器数据接口的MockApi模块.本篇文章就尝试为使用gradle的android项目设计实现Moc ...

  8. TODO:macOS编译PHP7.1

    TODO:macOS编译PHP7.1 本文主要介绍在macOS上编译PHP7.1,有兴趣的朋友可以去尝试一下. 1.下载PHP7.1源码,建议到PHP官网下载纯净到源码包php-7.1.0.tar.g ...

  9. In-Memory:在内存中创建临时表和表变量

    在Disk-Base数据库中,由于临时表和表变量的数据存储在tempdb中,如果系统频繁地创建和更新临时表和表变量,大量的IO操作集中在tempdb中,tempdb很可能成为系统性能的瓶颈.在SQL ...

  10. In-Memory:内存优化表的事务处理

    内存优化表(Memory-Optimized Table,简称MOT)使用乐观策略(optimistic approach)实现事务的并发控制,在读取MOT时,使用多行版本化(Multi-Row ve ...

随机推荐

  1. selenium 模块使用

    selenium 概念:基于浏览器自动化的一个模块,可以模拟浏览器行为 环境的安装:下载selenium模块 selenium和爬虫之间的关联是什么? 便捷的获取页面中动态加载的数据 requests ...

  2. 5. Git初始化及仓库创建和操作

    4. Git初始化及仓库创建和操作 基本信息设置 1. 设置用户名 git config --global user.name 'itcastphpgit1' 2. 设置用户名邮箱 git confi ...

  3. C++ | 虚函数表内存布局

    虚表指针 虚函数有个特点.存在虚函数的类会在类的数据成员中生成一个虚函数指针 vfptr,而vfptr 指向了一张表(简称,虚表).正是由于虚函数的这个特性,C++的多态才有了发生的可能. 其中虚函数 ...

  4. 原来 flexbox 是这么工作的

    Flexbox 是一种 CSS 布局机制,可以说是目前浏览器原生支持的最好.使用最广泛的布局机制了.本文通过一些例子来说明 Flexbox 布局的工作原理,可以让我们更好的使用 Flexbox. 与 ...

  5. 移动端比1px还小的border

    巧用border 在移动端 经常出现border,细边框但有的时候 产品大大1px甚至乎会觉得不够细那么要如何写出比1px还要小的border下面是代码 希望对大家有所帮助 .thinner-bord ...

  6. 谈谈关于CSS中transform属性之scale

    谈谈关于scale属性 scale是什么? 根据W3C定义 ,scale主要是进行缩放和转化: scale能做什么? 1.1px细线 <div class="wrap"> ...

  7. 《每周一点canvas动画》——3D点线与水波动画

    <每周一点canvas动画>--差分函数的妙用 每周一点canvas动画代码文件 好像上次更新还是十一前,这唰唰唰的就过去大半个月了,现在才更新实在不好意思.这次我们不涉及canvas 3 ...

  8. nextSibling和lastSibling

    在FireFox中包含众多空格作为文本节点,因此在我们使用nextSibling和previousSibling时就会出现问题.因为FireFox会把文本节点误当做元素节点的兄弟节点来处理.我们可以添 ...

  9. canvas离屏、旋转效果实践——旋转的雪花

    效果展示理论基础--"常见的canvas优化--模糊问题.旋转效果" 用离屏canvas画基础部分 1.封装画线函数 function drawLine(ctx,x1,y1,x2, ...

  10. java基础-多线程互斥锁

    多线程(JDK1.5的新特性互斥锁)* 1.同步 * 使用ReentrantLock类的lock()和unlock()方法进行同步* 2.通信 * 使用ReentrantLock类的newCondit ...