我们接触过STL中的部分容器,比如:vector,list,deque,forward_list(C++11)等,这些容器统称为序列式容器,因为其底层为线性序列的数据结构,里面存储的是元素本身。那么什么是关联式容器?它与序列式容器有什么区别?
关联式容器也是用来存储数据的,与序列式容器不同的是,其里面存储的是
用来表示具有一一对应关系的一种结构,该结构中一般只包含两个成员变量key和value,key代表键值,value表示与key对应的信息。比如:现在要建立一个英汉互译的字典,那该字典中必然有英文单词与其对应的中文含义,而且,英文单词与其中文含义是一一对应的关系,即通过该单词,在词典就可以找到与其对应的中文含义;比如每个学生的学号跟他的姓名,年龄,专业(类型可定义成数组)存在对应关系,找到学号,就可以查到这个学生的相关信息。
SGI-STL中关于键值对的定义:
//STL中关于键值对的定义
template<class T1,class T2>
struct pair
{
typedef T1 first_type;
typedef T2 first_type;
T1 first;
T2 second;
//构造函数初始化
pair()
:first(T1())//key
,second(T2())//value
{}
//拷贝构造函数,使用实例pair
pair(const T1& a,const T2& b)
:first(a)
,second(b)
{}
};
根据应用场景不同,STL总共实现了两种不同结构的关联式容器:树形结构与哈希结构。树形结构的关联式容器主要有四种:map,set,multimap,multiset。这四种容器的共同点是:使用平衡二叉树(即红黑树)作为底层,容器中的元素是一个有序的序列。下面依次介绍每一个容器。
1.set翻译为集合,是一个内部自动有序且不含重复元素的关联式容器。当我们想要去重,就可以利用到set,是一个很直观的接口,并且加入set之后可以实现自动排序,需要注意的是set的元素默认按照小于比较。
2、在set中,每个value必须是唯一的。set中的元素不能在容器中修改(元素总是const),但是可以在容器中插入或删除数据。一般的做法是先删除旧元素,然后添加新元素,这当然是为了维护里面元素的有序性。
3.set在底层使用二叉搜索树(红黑树)实现的。
4.与map/multimap不同,map/multimap中存储的是真正的键值对
5.在set中查找某个元素,时间复杂度为:log2N
//头文件
#include
std::set
template<class T,class Compare = less<T>,class Alloc = allocator<T>>
T:set中存放元素的类型,实际在底层存储
Compare(仿函数):set中元素默认按照小于来比较
Alloc:set中元素空间的管理方式,使用STL提供的空间配置器管理
点此进入->set的文档
我在下列代码一一解释了关于set常见的函数该如何使用,
#include
#include
using namespace std;
int main()
{
set<int> myset;//定义
//迭代器的使用
set<int>::iterator itlow, itup;
for (int i = 1; i < 10; i++)
{
//插入数据
//注意set会将元素自动排序
myset.insert(i * 10);//10 20 30 40 50 60 70 80 90
}
//删除35-75之间的数据
itlow = myset.lower_bound(35);//返回第一个大于等于key_value的定位器
itup = myset.upper_bound(75);//返回最后一个小于等于key_value的定位器
//注意:set中的删除操作不进行错误检查,所以用的时候最好用一下find()函数,
//返回给定值定位器,如果没找到则返回end()。
myset.erase(itlow, itup);
/*for (auto it = myset.begin(); it != myset.end(); ++it)
{
cout << *it <<' ';
}*/
//这里可以使用范围for,需要加&,因为拷贝也是存在一定代价的,不需要修改的话,也尽量加const
//范围for的底层是迭代器,操作由编译器完成
for (auto& s : myset)
{
cout << s << ' ';
}
cout << endl;
//count() 用来查找set中某个键值出现的次数。这个函数在set并不是很实用,在map中的用处比较大
//因为一个键值在set只可能出现0或1次,==这样就变成了判断某一键值是否在set出现过==。
cout << myset.count(100) << endl;//0不存在
cout << myset.count(30) << endl;//1存在
return 0;
}
点此进入->map的官方文档
1.map是关联式容器,它按照特定的次序(按照key来排序),存储由key和value组合而成的元素。
2.map中,是真正的键值对
typedef pair<const key,T> value_type
相比set,map多了一个参数
#include
std::map
template<class key,class T,class Compare = less<key>,class Alloc = allocator<pair<const Key,T>>
key: 键值对中key的类型
T: 键值对中value的类型
Compare: 比较器的类型,map中的元素是按照key来比较的,缺省情况下按照小于来比
较
Alloc:通过空间配置器来申请底层空间,不需要用户传递,除非用户不想使用标准库提供的
空间配置器
注意:在使用map时,需要包含头文件。
map的官方文档已包括所有关于map的接口函数,现在我们来尝试使用它的常用接口。
最重要的应该有:插入函数insert,[ ]下标访问操作符,对于pair的理解,迭代器的使用,删除函数。
#include
#include
#include
using namespace std;
int main()
{
//定义一个map对象
map<int, string> Stu;
//1.用insert函数插入pair
Stu.insert(pair<int, string>(01, "张三"));
Stu.insert(pair<int, string>(02, "李四"));
Stu.insert(pair<int, string>(03, "王五"));
//但是我们发现在这里,写一个pair很不方便,因此我们可以用make_pair
//make_pair的底层:是一个函数模板,还是调用pair去构造
/*template
pair make_pair(T1 x,T2 y)
{
return (pair(x, y));
}*/
Stu.insert(make_pair(04, "李芳"));
//3.第三种用[]的方式插入数据
Stu[123] = "王五";
//4.map的迭代器
//map::iterator it = Stu.begin();
auto it = Stu.begin();
while (it != Stu.end())
{
cout << it->first << ":" << it->second << endl;
++it;
}
//最好加上引用,如果不修改的话,把const加上更好
for (const auto& kv : Stu)
{
cout << kv.first << ":"<< kv.second << endl;
}
cout << endl;
return 0;
}
活学活用:统计水果出现的次数
//利用map,统计水果出现的次数
string arr[] = { "苹果", "西瓜", "香蕉", "草莓", "苹果", "西瓜", "苹果", "苹果", "西瓜", "苹果", "香蕉", "苹果", "香蕉" };
map<string, int> countMap;
for (auto& e: arr)
{
//map::iterator it = countMap.find(e);
auto it = countMap.find(e);//find函数找不到则返回end()
if (it == countMap.end())
{
countMap.insert(make_pair(e,1));
}
else
{
it->second++;
}
}
//有个更简洁的方法
//for (auto& e : arr)
//{
// countMap[e]++;
//}
for (const auto& e : countMap)
{
cout << e.first << ":" << e.second;
}
注意:multimap和map的唯一不同就是:map中的key是唯一的,而multimap中key是可以重复的。
multimap中的接口可以参考map,功能都是类似的。
注意: