标准库定义了两种主要的关联容器:map和set

map中的元素时一些关键字-值(key-value)对,关键字起到索引的作用,值则表示与索引相关的数据。set中每个元素只包含一个关键字,可以完成高效的关键字查询操作。

由map和set延伸出来一共8个关联容器,或者是一个map,或者是一个set,multi开头的是可以重复关键字的关联容器,unordered_开头的是无序关联容器。unordered_multi开头的是允许重复关键字的无序关联容器。

map和multimap定义在头文件<map>中,set和multiset定义在<set>中,4个无序容器分别定义在<unordered_map>和<unordered_set>中。

1.使用map:

map<string, size_t> words;
string word;
while (cin >> word)
++words[word];
for (const auto elmt : words)
cout << elmt.first << elmt.second << endl;

我们定义了一个map,关键字是string类型,值为size_t类型,每次从标准输入读一个单词,将这个单词的计数+1,如果words离不存在word,则创建一个word。

最后,打印出words内容,map的元素类型时pair,first成员表示关键字,second成员表示值。

2. 使用set

map<string, size_t> words;
set<string> exclude = {"The", "But", "And", "Or"}; string word;
while (cin >> word)
if (exclude.find(word) == exclude.end())
++words[word];

set保存了我们想要忽略的单词,在if判断里,如果word不在set exclude里,find成员函数返回尾后迭代器,并将这个单词的计数+1.

定义关联容器

定义一个map时,我们必须既指明关键字类型又指明值的类型,定义一个set时,只需指明关键字类型就可以,因为set的关键字就是值。

而初始化一个map时,必须提供关键字类型和值类型,我们将一对关键字-值放在一个花括号里: { key, value};

一个map或set的关键字是唯一的,multimap和multiset没有这个限制,初始化一个multimap或multiset时,可以重复关键字:

vector<int> vec;
for (vector<int>::size_type i = 0; i < 10; ++i){
// 每个关键字重复一次
vec.push_back(i);
vec.push_back(i);
} set<int> st(vec.begin(), vec.end());
multiset<int> mltst(vec.begin(), vec.end()); cout << vec.size() << endl; // 输出20
cout << st.size() << endl; // 输出10
cout << mltst.size() << endl; // 输出20
  • 关键字类型的要求(非常重要!)

关联容器对关键字的类型有一定的要求,先讲有序容器, 关键字类型必须定义元素的比较方法,默认情况下,标准库使用关键字类型的 < 运算符来比较关键字。

传递给排序算法的可调用对象必须满足与关联容器中关键字一样的类型要求。

可以向算法提供我们自己定义的比较操作,类似的,我们也可以提供自己定义的操作来代替关键字的 < 运算符。所提供的操作必须在关键字类型上定义一个严格弱序

它具备以下性质:

  • 两个关键字不能同时"小于等于"对方;如果 k1 "小于等于" k2,那么k2绝不能 "小于等于" k1.
  • 如果k1 "小于等于" k2, 且k2 "小于等于" k3, 那么k1必须 "小于等于" k3.
  • 如果存在两个关键字,任何一个都不 "小于等于" 另一个, 那么我们称这两个关键字是 "等价的"。如果 k1 "等价于" k2, k2 "等价于" k3,那么k1必须 "等价于" k3.

实际编程中,重要的是,如果一个类型定义了行为正常的 < 运算符,则它可以用作关键字类型。

我们定义一个类叫 Sales_data, 它又一个成员函数为compareIsbn, 用来比较它的成员isbn, 则我们可以用map来保存一个Sales_data类:

 bool compareIsbn(const Sales_data& lhs, const Sales_data& rhs) {
return lhs.isbn < rhs.isbn;
} map<Sales_data, decltype(compareIsbn)*> Smap(compareIsbn);

我们使用decltype来指出自定义操作的类型,当使用decltype获取一个函数指针类型时必须加上一个 * 来指出我们需要一个函数指针类型。我们使用compareIsbn来初始化Smap, 表示我们向Smap中添加元素时,根据compareIsbn来为元素排序。

pair类型

pair类型定义在头文件 <utility>中。一个pair保存两个数据成员,当创建一个pair时,我们需要指出两个类型名。

pair<T1, T2> p;             p是一个pair类型,对两个类型分别为T1和 T2的成员进行值初始化
pair<T1, T2> p(v1, v2); v1和v2分别对p的first和second成员进行初始化
pair<T1, T2> p = {v1, v2}; 等价于p(v1, v2) make_pair(v1, v2); 函数返回一个用v1和v2初始化的pair,pair的类型从v1和v2推断出来 p.first; 返回p的名为first(共有的)的数据成员
p.second; 返回p的名为second(共有的)的数据成员 p1 relop p2 关系运算符(< <= > >=)按字典序定义: 当p1.first < p2.first或 !(p2.first < p1.first) && p1.second < p2.second 成立时,p1 < p2 p1 == p2 当first和second成员分别相等时,两个pair相等。
p1 != p2

创建pair对象的函数:

假如一个函数返回pair类型,我们可以这样返回:

pair<string, int>
process(vector<string>& v) {
if (!v.empty())
return {v.back(), v.back().size()}; // 列表初始化
else
return pair<string, int>(); // 隐式构造返回值
}

除了这两种,我们还可以使用make_pair来生成pair对象:

if (!v.empty())
return make_pair(v.back(), v.back().size());

C++ Primer : 第十一章 : 关联容器之概述、有序关联容器关键字要求和pair类型的更多相关文章

  1. C++ Primer : 第十一章 : 关联容器之关联容器的迭代器和操作

    关联容器的操作 除了和顺序容器定义的类型之外,关联容器还定义了一下几种类型: 关联容器额外的类型别名  key_type    此容器类型的关键字类型 mapped_type  每个关键字关联的类型, ...

  2. C++Primer 第十一章

    //1.关键容器支持高效的关键字查找和访问. map 关联数组:保存关键字-值对.通过关键字来查找值. set 关键字即值,即只保存关键字的容器. multimap 关键字可重复出现的map mult ...

  3. C++ Primer : 第十一章 : 关联容器示例: 一个单词转换的map

    单词转换就是:将一些缩写的单词转换为实际的文本.第一个文件保存的是转换的规则,而第二个文件保存的是要转换的文本. 假设单词转换的规则的文件如下: brb be right back k okay? y ...

  4. 第十一章:使用Apriori算法进行关联分析

  5. 第十一章:random库概述

    random库是使用随机数的python标准库 伪随机数:采用梅森旋转算法生成的伪随机序列中的元素 random库主要用于产生随机数 使用random库:import random random库的概 ...

  6. C++ Primer 第3章 字符串、向量和数组

    C++ Primer 第3章 字符串.向量和数组 C Primer 第3章 字符串向量和数组 1 命名空间的using声明 2 标准库类型string 3 标准库类型vector 4 迭代器介绍 5 ...

  7. File类与常用IO流第十一章——打印流

    第十一章.打印流 概述:java.io.PrintStream extends OutputStream,为其他输出流添加了功能,使题目能够方便的打印各种数据值表示形式. 特点: 只负责数据的输出,不 ...

  8. 【C++ Primer 第11章】2. 关联容器操作

    练习答案 一.访问元素 关联容器额外类型别名  key_type 此容器类型的关键字类型 mapped_type 每个关键字关联的类型,只 适用于map mapped_type 对于set,与key_ ...

  9. [C++ Primer] : 第11章: 关联容器

    目录 使用关联容器 关联容器概述 关联容器操作 无序容器 使用关联容器 关联容器与顺序容器有着根本的不同: 关联容器中的元素是按关键字来保存和访问的, 按顺序容器中的元素是按它们在容器中的位置来顺序保 ...

随机推荐

  1. libpng交叉编译安装

    tar xzf libpng-1.5.22.tar.gz cd libpng-1.5.22 mkdir tmp 打开Makefile文件并修改CC=arm-linux-gcc ./configure ...

  2. Java对象的序列化和反序列化实践

    2013-12-20 14:58 对象序列化的目标是将对象保存在磁盘中,或者允许在网络中直接传输对象.对象序列化机制允许把内存中的Java对象转换成平台无关的二进制流,从而允许把这种二进制流持久的保存 ...

  3. 继承自NSObject的类不能用CGRect

    我用的是Xcode6.2. 系统默认没有pch文件. 所以没有自动导入UIKit包. 我在继承NSObject类里也不能用CGRect或者UI开头的控件,原因也是Xcode6.2以后版本 缺少UIKi ...

  4. office2013不能正常使用

    感觉非常卡一样 还不能显示内容 工具栏也点不动   这个应该是Office硬件图形加速惹得祸,解决办法: 文件>选项>高级>显示,找到“禁用硬件图形加速”在前面打勾,关闭再打开就OK ...

  5. iOS开发多线程篇—线程间的通信(转)

    这里转载 给自己一个备份 一.简单说明 线程间通信:在1个进程中,线程往往不是孤立存在的,多个线程之间需要经常进行通信 线程间通信的体现 1个线程传递数据给另1个线程 在1个线程中执行完特定任务后,转 ...

  6. ios创建bundle的图片资源文件(转)

    在ios开发中为了方便管理资源文件,可以使用bundle的方式来进行管理,比如kkgridview里就是把所需的图片文件全部放在一个bundle来管理的 . 切记目前iOS中只允许使用bundle管理 ...

  7. spring六种种依赖注入方式

    平常的java开发中,程序员在某个类中需要依赖其它类的方法,则通常是new一个依赖类再调用类实例的方法,这种开发存在的问题是new的类实例不好统一管理,spring提出了依赖注入的思想,即依赖类不由程 ...

  8. SVG 2D入门5 - 颜色的表示

    SVG和canvas中是一样的,都是使用标准的HTML/CSS中的颜色表示方法,这些颜色都可以用于fill和stroke属性.基本有下面这些定义颜色的方式:1. 颜色名字: 直接使用颜色名字red, ...

  9. windows-ubuntu环境变量的设置格式的不同

    1  在Ubuntu下输出环境变量,比如JAVA_HOME, 使用cat或者echo $JAVA_HOME即可,但是在windows下不可以, windows不支持cat命令,只能使用echo %JA ...

  10. Design Patterns---- Strategy 模式

    设计模式:可复用面向对象软件的基础 书中对 Strategy 模式的定义如下: 定义了一系列的算法,把它们一个个封装起来,并且使它们可相互替换.本模式使得算法可独立于它的用户而变化. 案例:设计一个商 ...