关于C++的STL库相信大家已经知道是什么了,也正是因为有了它才能让C++变得如此方便,今天我们就来看一下STL库中map与set的使用方法。
我们之前已经接触过STL中的部分容器,比如vector、list、deque等,这些容器称为序列式容器,因为其底层为线性的数据结构,里面存储的是元素本身,那么什么是关联式容器呢?
关联式容器也是用来存放数据的,与序列式容器不同的是,里面存储的是
用来表示具有一一对应关系的一种结构,该结构中一般只包含两个成员变量key和value,key代表键值,value表示与key对应的信息。
根据应用场景的不同,STL总共实现了两种不同结构的关联式容器:树型结构与哈希结构。树型结构的关联式容器主要有四种:map、set、multimap、multiset。这四种容器的共同点是:使用平衡搜索树(即红黑树)作为其底层结构,容器中的元素是一个有序的序列。下面我们会依次介绍每一个容器。
set<int> s; //注意一定指明存储元素的类型
set<int> s = { 0,2,5,1,4,0,6 };
int a[] = { 0,1,5,1,3,6,7 };
set<int> s(a, a + sizeof(a) / sizeof(int));
set<int> s = { 0,2,5,6,7 };
set<int> copy(s);
返回set中起始位置元素的迭代器
set<int> s = { 6,5,4,3,2,1 };
//插入数据后默认顺序为1 2 3 4 5 6
set<int>::iterator it = s.begin();
cout << *it << endl;
//最后输出的结果为1
返回set中最后一个元素的后面的迭代器
set<int> s = { 6,5,4,3,2,1 };
//插入数据后默认顺序为1 2 3 4 5 6
set<int>::iterator it = s.end();
cout << *(it - 1) << endl;
//最后输出的结果为6
后面的迭代器的用法与前面的用法相同,我们就不再举例了。
返回set中起始位置元素的const迭代器
返回set中最后一个元素后面的const迭代器
返回set第一个元素的反向迭代器,即set的最后一个元素
返回set中最后一个元素后面的反向迭代器
注意: 这里经过特殊的处理,下一个元素并不是我们所理解的第一个元素的下一个元素。
这样进行处理的话,我们的使用迭代器遍历数据的操作就和普通迭代器一致了,不同的迭代器以相同的做法达到相同的目的。后面遍历以迭代器遍历数据的时候我们还会提及。
返回set第一个元素的const反向迭代器,即set的最后一个元素
返回set中最后一个元素后面的const反向迭代器
set<int> s = { 1,0,9,3,2 };
for (auto e : s)
{
cout << e << " ";
}
//输出结果:0 1 2 3 9
注意: set的输出结果是有序的,这是因为它的底层结构为红黑树的原因,可以自动排序,自动去重,所以在set中是没有重复的元素的。
set天生 去重 + 排序
set<int> s = { 1,0,1,0,9,3,2,9,3 };
for (auto e : s)
{
cout << e << " ";
}
//最后的输出结果为: 0 1 2 3 9
set<int> s = { 6,5,4,3,2,1 };
set<int>::iterator it = s.begin();
while (it != s.end())
{
cout << *it << " ";
it++;
}
//1 2 3 4 5 6
cout << endl;
相信这个对于大家来说都没有问题
set<int> s = { 6,5,4,3,2,1 };
//1 2 3 4 5 6
set<int>::reverse_iterator it = s.rbegin();
while (it != s.rend())
{
cout << *it << " ";
it++;
}
//6 5 4 3 2 1
cout << endl;
但是不知道这个操作大家会怎么去理解。
错误思路
很多人会想rbegin()返回的就是最后一个元素的位置,rend()返回的是第一元素前一个位置,所以我要让it–,来实现遍历所有元素的目的,这样是不正确的,并且会报出错误。
正确思路
因为我们的语法操作要实现与使用begin(),end()时的一致,所以在这里it也是进行++,即it++,我们要把整个集合反过来看,即:6 5 4 3 2 1,反过来后rbegin()就可以理解为begin(),rend()就可以理解为end(),其他的操作与正式遍历一致,这样就很好理解了,it++也就不成问题。
这里涉及到了我们前面所学的仿函数less和greater,less是按照升序进行排序,greater是按照降序进行排序
1.set默认排升序,即less,我们在构造set时,如果没有给定仿函数,就以升序的形式输出数据
2.如果想排降序的话,就要在构造set对象时显示传仿函数
set<int,greater<int>> s = {0,1,2,3,4,5};
这样输出数据是就是降序
检测set是否为空,空返回true,否则返回false
返回set中有效元素的个数
insert()函数,只能用insert函数插入数据
set<int> s;
s.insert(4);
s.insert(2);
int main()
{
int a[] = { 0,1,5,1,3,6,7 };
set<int> s(a, a + sizeof(a) / sizeof(int));
set<int>::iterator it = s.begin();
while (it != s.end())
{
//*it = 10;//注意在set中无法赋值,否则会破坏树
cout << *it << " ";
it++;
}
//0 1 3 5 6 7
cout << endl;
//删除方法1:
s.erase(3);//可以直接删除
for (auto e : s)
{
cout << e << " ";
}
//0 1 5 6 7
cout << endl;
s.erase(30);//没有该元素,删除操作不会执行
for (auto e : s)
{
cout << e << " ";
}
//0 1 5 6 7
cout << endl;
return 0;
}
1.错误删除方式
int main()
{
int a[] = { 0,1,5,1,3,6,7 };
set<int> s(a, a + sizeof(a) / sizeof(int));
set<int>::iterator it = s.begin();
while (it != s.end())
{
cout << *it << " ";
it++;
}
//0 1 3 5 6 7
cout << endl;
//通过迭代器删除数据
//但是以这种方式查找没有的数据的位置时,返回的是end(),后面删除就会出错。所以不能直接删除
set<int>::iterator it1 = s.find(20);
//不进行判断,直接对找到的位置进行删除
s.erase(it1);
for (auto e : s)
{
cout << e << " ";
}
//0 1 3 5 6 7
cout << endl;
return 0;
}
如果不加判断条件就会报出这样的错误。
2.正确删除方式
int main()
{
int a[] = { 0,1,5,1,3,6,7 };
set<int> s(a, a + sizeof(a) / sizeof(int));
set<int>::iterator it = s.begin();
while (it != s.end())
{
cout << *it << " ";
it++;
}
//0 1 3 5 6 7
cout << endl;
set<int>::iterator it1 = s.find(20);
if (it1 != s.end())//这里要先要判断一下是否有该数据
{
//存在该数据就会执行删除操作
s.erase(it1);
}
for (auto e : s)
{
cout << e << " ";
}
//0 1 3 5 6 7
cout << endl;
return 0;
}
为什么第一种直接删除操作没有要删除的元素时不报错,那是因为直接删除操作就类似于我们所说的正确的迭代器删除元素的方法,相当于自己加了if判断语句,所以才没有报出错误。
既然已经有set容器了,那么multset又是用来干什么的呢?
multset允许键值冗余,也就是里面可以存相同的数据元素。在其他的一些函数和操作方面multiset与set是一致的,下面我们直接看他比较特殊的性质
直接通过代码和测试结果查看
int main()
{
//multiset允许键值冗余
int a[] = { 3,1,2,1,6,3,8,3,5,3 };
multiset<int> s(a, a + sizeof(a) / sizeof(int));
//排序但不去重
for (auto e : s)
{
cout << e << " ";
}
cout << endl;
cout << s.count(1) << endl;//输出2,有两个1
auto pos = s.find(3);//如果有多个值,返回中序的第一个
while (pos != s.end())
{
cout << *pos << " ";
++pos;
}
cout << endl;
s.erase(3);
for (auto e : s)
{
cout << e << " ";
}
cout << endl;
pos = s.find(1);//找到第一个1的位置,然后将第一个1删除
if (pos != s.end())
{
s.erase(pos);
}
for (auto e : s)
{
cout << e << " ";
}
return 0;
}
通过上面的代码和结果的分析,我们就可以很好的了解multiset的使用了。
构造空的map
map<string,string> dict;
pair
在map中插入键值对,注意x是一个键值对,返回值也是键值对:iterator代表插入元素的位置,bool代表是否插入成功,这里insert函数的设计与后面map可以使用operator[]有密切关系
map<string, string> dict;//创建一个键值对为类型的map
//插入方式一
pair<string,string> kvl("sort","排序");//创建一个结构体,隐式类型转换
dict.insert(kvl);//可以直接将kvl插入
//也可以使用下面的匿名对象将值插入dict对象中
dict.insert(pair<string, string>("test", "测试"));
//插入方式二
//如果感觉pair太长就可以将其进行typedef,但是这个其实也不方便,毕竟还要typedef一下
typedef pair<string, string> DictKV;
dict.insert(DictKV("string", "字符串"));
插入方式三(平时都习惯这么定义)
dict.insert(make_pair("left", "左边"));
1.size_type erase(const key_type& x)
删除键值为x的元素
map<string, string> dict;
dict.insert(make_pair("left", "左边"));
dict.insert(make_pair("right", "右边"));
dict.insert(make_pair("sort", "排序"));
dict.insert(make_pair("string", "字符串"));
//这里插入元素以后会按照属于key的比较方式进行排序
dict.erase("left");//删除key为left的元素
dict.erase("string");//删除key为string的元素
2.void erase(iterator position)
删除position位置上的元素
map<string, string> dict;
dict.insert(make_pair("left", "左边"));
dict.insert(make_pair("right", "右边"));
dict.insert(make_pair("sort", "排序"));
dict.insert(make_pair("string", "字符串"));
map<string, string>::iterator it = dict.find("left");
dict.erase(it);//删除it位置的元素
3.void erase(iterator first,iterator last)
删除[first,last)区间中的元素
map<string, string> dict;
dict.insert(make_pair("left", "左边"));
dict.insert(make_pair("right", "右边"));
dict.insert(make_pair("sort", "排序"));
dict.insert(make_pair("string", "字符串"));
dict.erase(dict.begin(),dict.end());//直接删除全部的元素
void swap (map& x)
交换两个map中的元素
map<string, string> dict1;
dict1.insert(make_pair("left", "左边"));
dict1.insert(make_pair("right", "右边"));
map<string, string> dict2;
dict2.insert(make_pair("sort", "排序"));
dict2.insert(make_pair("string", "字符串"));
//交换两个map中的元素
dict1.swap(dict2);
void clear()
将map中的元素清空
iterator find(const key_type& x)
在map中查找key为x的元素,找到返回该元素的位置的迭代器,否则返回end()
size_type count(const key_type& x) const
返回key为x的键值在map中的个数,注意map中key是唯一的,因此该函数的返回值要么为0,要么为1,因此也可以用该函数来
检测一个key是否在map中。
begin:首元素的位置,end:最后一个元素的下一个位置
与begin和end意义相同,但cbegin和cend所指向的元素不能修改
反向迭代器,rbegin在end位置,rend在begin位置,其
++和–操作与begin和end操作移动相反
与rbegin和rend位置相同,操作相同,但crbegin和crend所
指向的元素不能修改
以迭代器的方式遍历map中的元素
map<string, string> dict;
dict.insert(make_pair("left", "左边"));
dict.insert(make_pair("right", "右边"));
dict.insert(make_pair("sort", "排序"));
dict.insert(make_pair("string", "字符串"));
map<string, string>::iterator it = dict.begin();
//元素访问方式一(但是一般不这样使用)
while (it != dict.end())
{
cout << (*it).first << ":" << (*it).second << endl;
//*it为结构体,然后 以结构体.成员变量的形式访问
it++;
}
cout << endl;
//元素访问方式二(一般使用方式)
it = dict.begin();
while (it != dict.end())
{
cout << it->first << ":" << it->second << endl;
//it->first 其实是 it->->first(it->返回的是结构体指针,但是这里编译器进行了优化)
it++;
}
cout << endl;
bool empty() const
检测map中的元素是否为空,是返回true,否则返回false
size_type size() const
返回map中有效元素的个数
mapped_type& operator[] (const key_type& k)
返回key对应的value
map的精髓就在operator[]这里,下面我们一起来看看他的功能,他远远没有表面看起来这么简单。
先来看一下该函数的介绍
下面我们来看看插入数据时的几种情景
上面的这几种情景是如何产生的呢,其实与该函数介绍中的最后一行有关,我将他单独拿了出来
这里可以看到该函数与insert函数有关系,我们还要再分析insert函数
下面我们分析整个函数流程
最后我们来看一下这里的[]有什么用处,或者说是他的应用场景,下面我会就举一个例子,但是用不同的方法实现,将二者对比一下
例:给一个水果的数组,我们要统计该数组中各种水果出现的次数
string a[] = { “苹果”,“香蕉”, “西瓜”, “西瓜”, “苹果”, “苹果”, “苹果”, “香蕉”, “苹果” };
string a[] = { "苹果","香蕉", "西瓜", "西瓜", "苹果", "苹果", "苹果", "香蕉", "苹果" };
map<string, int> CountMap;
for (auto& str : a)
{
map<string, int>::iterator it = CountMap.find(str);
if (it != CountMap.end())
{
it->second++;
}
else
{
CountMap.insert(make_pair(str, 1));
}
}
map<string, int>::iterator it1 = CountMap.begin();
while (it1 != CountMap.end())
{
cout << it1->first << ":" << it1->second << endl;
it1++;
}
最后的输出结果:
上面的代码看上去还可以,但是有了operator[]后还有更简洁的代码,实现起来更加方便。
string a[] = { "苹果","香蕉", "西瓜", "西瓜", "苹果", "苹果", "苹果", "香蕉", "苹果" };
map<string, int> CountMap;
for (auto& e : a)
{
//1.e不在CountMap中,插入pair(str,int()),然后对返回的次数++
//2.str在CountMap中,返回value(次数)的引用,次数++
CountMap[e]++;
}
map<string, int>::iterator it = CountMap.begin();
while (it != CountMap.end())
{
cout << it->first << ":" << it->second << endl;
it++;
}
最后的输出结果:
我们可以看到第二种代码实现起来更加方便简洁,这就是operator[]的作用。
注意: 在元素访问时,有一个与operator[]类似的操作at()(该函数不常用)函数,都是通过key找到与key对应的value然后返回其引用,不同的是:当key不存在时,operator[]用默认value与key构造键值对然后插入,返回该默认value,at()函数直接抛异常。
multimap与map的函数的作用都是相似的,就和multiset与set之间的关系一样,map中的key是唯一的,而multimap中key是可以重复的。并且在multimap中没有重载operator[]操作, 因为在该容器中key的值是可以重复的,这也就说明一个key值可能会对应多个value,所以不会实现该函数。
关于C++中的map与set我们就讲解到这里了,map与set都是很重要的容器,大家下来后一定要自己动手去敲一敲代码,加深一下印象,同时也有利于后面的使用,如果文章中有错误大家可以在评论区指出,我看到后一定会第一时间修改,最后如果你觉得本章内容对你有用的话,就给一波三连吧,你们的支持就是我写博客最大的动力。