multimap 多重映照容器
multimap 與 map 一樣,都是使用紅黑樹對記錄型的元素數據,按元素鍵值的比較關系,進行快速的插入、刪除和檢索操作,所不同的是 multimap 允許將具有重復鍵值的元素插入容器。在 multimap 容器中,元素的鍵值與元素的映照數據的映照關系,是多對多的,因此,multimap 稱為多重映照容器。multimap 與 map 之間的多重特性差異,類似於 multiset 與 set 的多重特性差異。
multimap 多重映照容器實現了 Sorted Associative Container 、Pair Associative Container 和 Multimap Associative Container 概念的接口規范。
multimap 與 map 容器有很多地方相似,這里不再重復介紹一些概念和函數,很多內容可以參考 map 里面的介紹。這里只講一些常用用法。看代碼。
multimap 與 map 容器共用同一個 C++ 標准頭文件 map 。在用到 multimap 的時候,需要用到宏語句 "#include <map>"

1 /* 不同於 map 容器,multimap 容器只能采用迭代器的方式,而不能用數組方式,遍歷容器的元素。迭代器方式的元素訪問,一般要用 begin 和 end 函數找出遍歷開始的首元素和結束元素,然后通過迭代器的 “++” 和 “*“ 操作取出元素。 2 下面的示例程序,將水果價格元素表的記錄打印出來。由於水果價格作為元素的鍵值,並采用默認的 less<float> 作比較函數對象,將按價格由小到大打印出來。 3 */ 4 5 -------------------------------------------------------- 遍歷 multimap 容器元素 6 #pragma warning(disable:4786) 7 #include <map> 8 #include <iostream> 9 using namespace std; 10 int main() 11 { 12 multimap<float, char*> mm; 13 mm.insert(pair<float, char*>(3.0f, "apple")); 14 mm.insert(pair<float, char*>(3.0f, "pear")); 15 mm.insert(pair<float, char*>(2.6f, "orange")); 16 mm.insert(pair<float, char*>(1.8f, "banana")); 17 mm.insert(pair<float, char*>(6.3f, "lichee")); 18 // 遍歷打印 19 multimap<float, char*>::iterator i, iend; 20 iend = mm.end(); 21 for (i=mm.begin(); i!=iend; ++i) 22 cout << (*i).second << ' ' << (*i).first << "元/斤 \n"; 23 24 return 0; 25 }

/* 利用 multimap 容器定義的反向迭代器 reverse_iterator 和 const_reverse_iterator ,及獲取反向首尾元素的 rbegin 和 rend 函數,可反向遍歷 multimap 容器的元素。 例如下面使用反向遍歷,將水果價格表的記錄按價格大小,由大到小打印出來。 注意:鍵值相同的 pear 將先於 apple 打印出來。可見,在等鍵值的反向遍歷情形下,后插入的元素先打印出來。 */ -------------------------------------------------------- 反向遍歷 multimap 容器的元素 #pragma warning(disable:4786) #include <map> #include <iostream> using namespace std; int main() { multimap<float, char*> mm; mm.insert(pair<float, char*>(3.0f, "apple")); mm.insert(pair<float, char*>(3.0f, "pear")); mm.insert(pair<float, char*>(2.6f, "orange")); mm.insert(pair<float, char*>(1.8f, "banana")); mm.insert(pair<float, char*>(6.3f, "lichee")); // 反向遍歷打印 multimap<float, char*>::reverse_iterator r_i, r_iend; r_iend = mm.rend(); for (r_i=mm.rbegin(); r_i!=r_iend; ++r_i) cout << (*r_i).second << ' ' << (*r_i).first << "元/斤 \n"; return 0; }

1 /* 由於鍵值允許重復插入,在 multimap 容器中具有同一個鍵值的元素有可能不只一個。因此,multimap 容器的 find 函數將返回第一個搜索到的元素位置,如果元素不存在,則返回 end 結束元素位置。equal_range 函數則返回一個可指示相等元素范圍區間的 pair 對象。 2 */ 3 4 -------------------------------------------------------- multimap 容器的元素的搜索1 5 #pragma warning(disable:4786) 6 #include <map> 7 #include <iostream> 8 using namespace std; 9 10 // 課程記錄結構體 11 struct CourseRecord 12 { 13 // 課程信息結構體 14 struct CourseInfo 15 { 16 char* course; // 課程名 17 int period; // 學時 18 char* required; // 必修或選修 19 }; 20 21 char* teacher; // 任課教師 22 CourseInfo cf; 23 CourseRecord(char* teacher_, char* course_, int period_, char* required_) 24 { 25 teacher = teacher_; 26 cf.course = course_; 27 cf.period = period_; 28 cf.required = required_; 29 } 30 }; 31 32 int main() 33 { 34 // 創建 multimap 容器對象 mm 35 typedef multimap<char*, CourseRecord::CourseInfo> coursemmap; 36 coursemmap mm; 37 // 插入第1條記錄 38 CourseRecord course1 = CourseRecord("張三","操作系統開發",60,"必修"); 39 pair<char*, CourseRecord::CourseInfo> pairCourse1(course1.teacher, course1.cf); 40 mm.insert(pairCourse1); 41 42 // 插入第2條記錄 43 CourseRecord course2 = CourseRecord("李四","編譯器開發",30,"必修"); 44 pair<char*, CourseRecord::CourseInfo> pairCourse2(course2.teacher, course2.cf); 45 mm.insert(pairCourse2); 46 47 48 // 插入第3條記錄 49 CourseRecord course3 = CourseRecord("李四","數據結構",20,"必修"); 50 pair<char*, CourseRecord::CourseInfo> pairCourse3(course3.teacher, course3.cf); 51 mm.insert(pairCourse3); 52 53 54 // 插入第4條記錄 55 CourseRecord course4 = CourseRecord("李四","Java 開發應用",38,"必修"); 56 pair<char*, CourseRecord::CourseInfo> pairCourse4(course4.teacher, course4.cf); 57 mm.insert(pairCourse4); 58 59 // 插入第5條記錄 60 CourseRecord course5 = CourseRecord("王五","C++程序設計",58,"必修"); 61 pair<char*, CourseRecord::CourseInfo> pairCourse5(course5.teacher, course5.cf); 62 mm.insert(pairCourse5); 63 64 // 記錄搜索 65 cout << "搜索<李四老師>的任課記錄:\n"; 66 pair<coursemmap::iterator, coursemmap::iterator> p = mm.equal_range("李四"); 67 // 打印結果 68 coursemmap::iterator i; 69 for (i=p.first; i!=p.second; ++i) 70 cout << (*i).first << ' ' 71 << (*i).second.course << ' ' 72 << (*i).second.period << "學時 " 73 << (*i).second.required << ' ' 74 << endl; 75 cout << endl << endl; 76 77 return 0; 78 }

1 -------------------------------------------------------- multimap 容器的元素的搜索2 2 #pragma warning(disable:4786) 3 #include <iostream> 4 #include <string> 5 #include <UTILITY> 6 #include <map> 7 using namespace std; 8 9 struct userdevice 10 { 11 string m_devicename; 12 long m_deviced; 13 int m_devicePopedom; 14 }; 15 16 typedef multimap<string, userdevice> USERTABLE; 17 typedef USERTABLE::const_iterator CIT; 18 typedef pair<CIT, CIT> Range; 19 20 int main() 21 { 22 // 定義一個迭代器 23 CIT it; 24 // 定義4個設備 25 userdevice d1, d2, d3, d4; 26 d1.m_deviced = 12341234; 27 d1.m_devicename = "d1"; 28 d1.m_devicePopedom = 123; 29 30 d2.m_deviced = 23622344; 31 d2.m_devicename = "d2"; 32 d2.m_devicePopedom = 234; 33 34 d3.m_deviced = 3451234; 35 d3.m_devicename = "d3"; 36 d3.m_devicePopedom = 345; 37 38 d4.m_deviced = 43622344; 39 d4.m_devicename = "d4"; 40 d4.m_devicePopedom = 456; 41 42 USERTABLE m_user; 43 m_user.insert(pair<string, userdevice>("zhangsanfeng",d1)); 44 m_user.insert(pair<string, userdevice>("zhangsanfeng",d2)); 45 46 m_user.insert(pair<string, userdevice>("zhangsanfeng2",d3)); 47 m_user.insert(pair<string, userdevice>("zhangsanfeng2",d4)); 48 49 // 查找方法一(查找key值是"zhangsanfeng") 50 Range range = m_user.equal_range("zhangsanfeng"); 51 for (CIT i = range.first; i!=range.second; ++i) 52 { 53 cout<< i->second.m_deviced <<',' 54 << i->second.m_devicename.c_str()<<',' 55 << i->second.m_devicePopedom 56 << endl; 57 } 58 cout<< endl; 59 60 61 62 // 查找方法二(查找"zhangsanfeng2") 63 CIT it2 = m_user.find("zhangsanfeng2"); 64 while(it2!=m_user.end()) 65 { 66 cout<< it2->second.m_deviced <<',' 67 << it2->second.m_devicename.c_str()<<',' 68 << it2->second.m_devicePopedom 69 << endl; 70 ++it2; 71 } 72 cout<< endl; 73 74 75 // 遍歷 76 CIT it3 = m_user.begin(); 77 while (it3!=m_user.end()) 78 { 79 cout<< i->second.m_deviced <<',' 80 << it3->second.m_devicename.c_str()<<',' 81 << it3->second.m_devicePopedom 82 << endl; 83 84 ++it3; 85 } 86 cout<< endl; 87 88 return 0; 89 }

1 /* 下面的示例程序調用 multimap 容器的 size 和 count 函數,對元素個數和去某鍵值的元素個數進行統計 2 */ 3 4 -------------------------------------------------------- multimap 容器元素的設計 5 #pragma warning(disable:4786) 6 #include <map> 7 #include <iostream> 8 using namespace std; 9 10 11 int main() 12 { 13 multimap<int, char> mm; 14 // 打印元素個數:0 15 cout << mm.size() << endl; 16 mm.insert(pair<int, char>(3, 'a')); 17 mm.insert(pair<int, char>(3, 'c')); 18 mm.insert(pair<int, char>(4, 'd')); 19 mm.insert(pair<int, char>(5, 'e')); 20 // 打印鍵值為 3 的元素個數 21 cout << mm.count(3) << endl; 22 // 打印元素個數 23 cout << mm.size() << endl; 24 25 return 0; 26 }
---------------------- multimap 小結
multimap 多重映照容器也是一種依元素鍵值進行插入、刪除和檢索的有序關聯容器,元素的檢索具有對數階的算法時間復雜度。與 map 容器的區別是,multimap 允許元素鍵值出現重復,這導致 multimap 多重映照容器不能像 map 容器那樣,可簡便地利用數組方式訪問或添加容器元素。
multimap 缺點:和 map 的缺點差不多。multimap 是比較復雜的用法,如果能用 vector 、list 這些容器完成的事情,就不需要用到 multimap .
multimap 優點:相比 map ,multimap 能允許重復鍵值的元素插入。