STL:unordered_map使用笔记
参考网址:
- cpluscplus
- unordered_map与map的区别(CSDN)
1.概述
unordered_map的模板定义如下:
template < class Key,                                    // unordered_map::key_type
           class T,                                      // unordered_map::mapped_type
           class Hash = hash<Key>,                       // unordered_map::hasher
           class Pred = equal_to<Key>,                   // unordered_map::key_equal
           class Alloc = allocator< pair<const Key,T> >  // unordered_map::allocator_type
           > class unordered_map;
unordered_map是一种关联容器,存储基于键值和映射组成的元素,即key-value。允许基于键快速查找元素。在unordered_map中,键值唯一标识元素,映射的值是一个与该对象关联的内容的对象。
对于有序和无序性:
- unordered_map的无序体现在内部存储结构为哈希表,以便通过键值快速访问元素。
- 与之对应的有序的关联容器为map,map的有序体现在内部存储结构为红黑树,存储时元素自动按照从小到大的顺序排列。
而内部存储结构也决定了unordered_map和map在某些特性上的不同:
- 查找的效率 
  - unordered_map查找效率更高,可以达到O(1),但是对于元素子集的范围迭代效率较低。
- 对于map,按照中序遍历的遍历次序,能够方便迭代得出从小到大的元素
 
- unordered_map查找效率更高,可以达到
无序映射实现了直接访问操作符(operator[]),该操作符允许使用其键值作为参数直接访问映射值。容器中的迭代器至少是前向迭代器forward iterators。
2.属性
2.1 关联性
关联容器中的元素由他们的键引用,而不是由他们在容器中的绝对位置引用。
2.2 无序性
无序容器使用散列表来组织它们的元素,散列表允许通过它们的键快速访问元素。
2.3 Map映射
每个元素将一个键key与一个映射值value相关联:键意味着标识其主要内容是映射值的元素。
2.4 key的唯一性
在容器中没有两个元素有相同的key
2.5 Allocator-aware
容器使用一个分配器对象来动态地处理它的存储需求。
3.模板参数
template < class Key,                                    // unordered_map::key_type
           class T,                                      // unordered_map::mapped_type
           class Hash = hash<Key>,                       // unordered_map::hasher
           class Pred = equal_to<Key>,                   // unordered_map::key_equal
           class Alloc = allocator< pair<const Key,T> >  // unordered_map::allocator_type
           > class unordered_map;
-  key 键值的类型。unordered_map中的每个元素都是由其键值唯一标识的。 
-  T 映射值的类型。unordered_map中的每个元素都用来存储一些数据作为其映射值。 
-  Hash 一种一元函数对象类型,它接受一个key类型的对象作为参数,并根据该对象返回size_t类型的唯一值。这可以是一个实现函数调用操作符的类,也可以是一个指向函数的指针(参见构造函数)。默认为 hash<Key>。
-  Pred 接受两个键类型参数并返回bool类型的二进制谓词。表达式pred (a, b), pred是这种类型的一个对象,a和b是键值,返回true,如果是应考虑相当于b。这可以是一个类实现一个函数调用操作符或指向函数的指针(见构造函数为例)。这默认为equal_to<Key>,它返回与应用相等操作符(a==b)相同的结果。 
-  Allloc 用于定义存储分配模型的allocator对象的类型。默认情况下,使用allocator类模板,它定义了最简单的内存分配模型,并且与值无关。 
4.成员函数
4.1 构造函数与初始化
(1)模板类的默认构造函数,创建空的unordered_map
unordered_map<int, string> umap;
(2)使用初始化列表初始化
unordered_map<int, string> umap = unordered_map<int, string>({{1,"a"},{2,"b"}});  // 显式调用C++的构造函数
unordered_map<int, string> umap2({{3,"c"},{4,"d"}});	// 隐式调用构造函数,更简洁
unordered_map<string, string> umap{
    {"淘宝","https://www.taobao.com/"},
    {"京东","https://www.jd.com/"},
    {"天猫商城","https://jx.tmall.com/"} };
(3)拷贝构造函数初始化
// 拷贝构造函数 
unordered_map<int, string> umap4(umap3);
(4)迭代器初始化
// range
unordered_map<int, string> umap5(umap1.begin(), umap1.end());
(5)拷贝初始化
typedef std::unordered_map<std::string,std::string> stringmap;
first = {{"AAPL","Apple"},{"MSFT","Microsoft"}};  // init list
second = {{"GOOG","Google"},{"ORCL","Oracle"}};   // init list
third = merge(first,second);                      // move
first = third;                                    // copy	
4.2 capacity
(1)是否为空
cout << "first " << (first.empty() ? "is empty" : "is not empty") << endl;
cout << "first " << (second.empty() ? "is empty" : "is not empty") << endl;
(2)目前容量
cout << "thrid.size is" << third.size() << endl;
4.3 迭代
/**  Iteration **/
for (auto it = first.begin(); it != first.end(); it++){
    cout << " " << it->first << ":" << it->second;
}
cout<<endl;
4.4 元素的访问
(1)operator[]
first["GOOG"] = "Google";		// new element inserted
first["AAPL"] = "Apple";		// new element inserted
first["MSFT"] = "Microsoft";	// new element inserted
first["BOB"] = "Bob";
string brand1 = first["GOOG"];	// read
first["BOB"] = "";				// writen
for (auto it = first.begin(); it != first.end(); it++){
    cout << " " << it->first << ":" << it->second;
}
cout<<endl;
(2)at
unrdered_map<string,int> mymap = {
    {"Mars", 3000},
    {"Saturn", 60000},
    {"Jupiter", 70000}};
mymap.at("Mars") = 3396;
mymap.at("Saturn") += 127;
mymap.at("Jupiter") = mymap.at("Saturn") + 9638;
	
for (auto& x: mymap) {
    std::cout << x.first << ": " << x.second << std::endl;
}
4.5 元素的查找
(1)find
find函数的原型如下:
iterator find ( const key_type& k );
const_iterator find ( const key_type& k ) const;
find函数可以用来获取元素的迭代器:
std::unordered_map<std::string,double> mymap1 = {
	    {"mom",5.4},
	    {"dad",6.1},
	    {"bro",5.9} };
	
string person = "dad";
unordered_map<std::string,double>::const_iterator got = mymap1.find(person);
if(got == mymap1.end()){
    cout << "not found" << endl;
}else{
    cout << got->first << " is " << got->second << endl; 
}
注意迭代器的声明方式:
unordered_map<string,double>::const_iterator
(2)count
成员函数count判断集合中有没有键值k,函数原型如下:
size_type count ( const key_type& k ) const;
其中size_type是一个整型变量,如果无序map中有key k,那么返回值为1,否则返回值为0。
4.6 修改
(1)元素的插入
-  operator[] 使用操作符 map_name [key_name] = value,实现对应key的value值的覆盖,若是key值是原来不存在的key,那么实现了新的元素的插入。
-  emplace()在unordered_map中插入一个新元素(如果其键是唯一的)。此新元素是使用 args 作为元素构造函数的参数来构造的。 仅当容器中没有元素具有与要放置的元素等效的键(unordered_map中的键是唯一的)时,才会进行插入。 如果插入,这实际上会将容器大小增加一个。 存在类似的成员函数 insert,它将现有对象复制或移动到容器中。 // unordered_map::emplace #include <iostream> #include <string> #include <unordered_map> int main () { std::unordered_map<std::string,std::string> mymap; mymap.emplace ("NCC-1701", "J.T. Kirk"); mymap.emplace ("NCC-1701-D", "J.L. Picard"); mymap.emplace ("NCC-74656", "K. Janeway"); std::cout << "mymap contains:" << std::endl; for (auto& x: mymap) std::cout << x.first << ": " << x.second << std::endl; std::cout << std::endl; return 0; }
-  insert()insert的插入和emplace的插入类似,只有当键是唯一时,才能进行插入,同时size增加。 insert的插入形式比emplace更加灵活: (1) pair<iterator,bool> insert ( const value_type& val ); (2) template <class P> pair<iterator,bool> insert ( P&& val ); (3) iterator insert ( const_iterator hint, const value_type& val ); (4) template <class P> iterator insert ( const_iterator hint, P&& val ); (5) template <class InputIterator> void insert ( InputIterator first, InputIterator last ); (6) void insert ( initializer_list<value_type> il );/** 元素的插入 **/ // insert unordered_map<string,double> myrecipe, mypantry = {{"milk",2.0},{"flour",1.5}};; pair<string,double> myshopping ("baking powder",0.3); myrecipe.insert(myshopping); // copy insertion myrecipe.insert(mypantry.begin(), mypantry.end()); // range inseration myrecipe.insert({{"sugar",0.8},{"salt",0.1},{"sugar",0.9}}); // initializer list inseration cout << "myrecipe contains:" << endl; for(auto& x:myrecipe){ cout << x.first << ":" << x.second << " "; } 
(2)元素的删除
元素的删除使用erase()方法,删除的形式有多种:
//erase
std::unordered_map<std::string,std::string> mymap3;
// populating container:
mymap3["U.S."] = "Washington";
mymap3["U.K."] = "London";
mymap3["France"] = "Paris";
mymap3["Russia"] = "Moscow";
mymap3["China"] = "Beijing";
mymap3["Germany"] = "Berlin";
mymap3["Japan"] = "Tokyo";
cout << endl;
cout << "------------------------------------------" << endl;
// 根据位置,删除第一个:
mymap3.erase(mymap3.begin());
// 根据key
mymap3.erase("France");
// range
mymap3.erase(mymap3.find("Germany"), mymap3.end());
for (auto& x : mymap3) {
    cout << x.first << ":" << x.second << " ";
}










