C++STL——map

一、相关定义

map

  • 关联容器,存储相结合形成的一个关键值和映射值的元素
  • 提供一对一(第一个可以称为关键字,每个关键字只能在map中出现一次,第二个可以称为该关键字的值)的数据处理能力
  • map对象是模板类,需要关键字和存储对象两个模板参数

特征

  • Map 是一种Pair Associative Container,意味着它的值类型为 pair<const Key, Data>,而且也是 Unique Associative Container,也就是任何两个元素没有相同的key值;
  • map内部自建一颗红黑树(一种非严格意义上的平衡二叉树),这颗树具有对数据自动排序的功能,所以在map内部所有的数据都是有序的;
  • 在map对象中插入一个新元素不指向现有元素的迭代器失效。从map上删除一个元素,也没有任何迭代器失效,除非,当然,实际上指向正在被删除的元素的迭代器;
  • 增加和删除节点对迭代器的影响很小,除了那个操作节点,对其他的节点都没有什么影响;
  • map主要建立了key到value的映射。对于迭代器来说,可以修改实值,而不能修改key。

数据映射

举例说明:比如一个班级中,每个学生的学号跟他的姓名就存在着一一映射的关系,这个模型用map可以轻易描述,很明显学号用int描述,姓名用字符串描述

给出map描述代码:map<int,string> mapStudent;

二、map

【构造方法】

  • map<key,value> myMap;  //为了使用方便,可以对模板类进行一下类型定义  typedef map<int, string> myMap; myMAP enumMap;
  • map<key,value>::iterator iter;  //迭代器iter

【功能】

  • 自动建立Key-value的对应,key 和 value可以是任意你需要的类型。
  • 根据key值快速查找记录,查找的复杂度基本是Log(N),如果有1000个记录,最多查找10次,1000000个记录,最多查找20次。
  • 快速插入Key - Value 记录。
  • 快速删除记录
  • 根据Key 修改value记录。
  • 遍历所有记录

【常用方法】

insert方法

在map中插入一个元素,map中记录的元素通常为键值对,所以,在存储时会把键和值封装成pair然后进行插入,例如:phone.insert(pair<string,string>(name,number));其中name和number为string类型的变量。当然也可以简单的写成phone[name]=number; 此处phone即为map<string,string>类型的变量。因为map在实现过程中对[]进行了重载

第一种方式若插入的元素的键值已经存在于map中,那么就会插入失败,不会修改元素的键值对信息,若键值在map中查找不到,那么就会将该新元素加入到map中去。

第二种方式比较直观,但存在一个性能的问题。插入2时,先在phone中查找主键为name的项,没发现,然后将一个新的对象插入phone,键是name,值是一个空字符串,插入完成后,将字符串赋为number, 该方法会将每个值都赋为缺省值,然后再赋为显示的值,如果元素是类对象,则开销比较大。若找到键值为name的项,则用number更改原来的number值。

erease方法

erease主要是删除map中的某个项,需要参数key,例如phone.erease(name);此句的意思就是删除key值为name的键值对。

size方法

统计map中键值对的个数,phone.size()返回值即为phone中键值对的个数,若map为空则返回0

count方法

统计map中某个键值出现的次数,因为map中键值唯一,所以此方法可以用来检测某键值是否存在,例如在删除时可以phone.count(name),若为0则可以提示用户此键值不存在,若为1则直接删除。Ps:erease无论要删除的键值对是否存在都能正常执行。

begin/end方法

begin方法返回map迭代器类型,通过此迭代器与end方法的返回值进行比较就可以很容易的对map进行遍历。

clear方法

清空map中的所有元素 

empty方法

判断map是否为空,若为空则返回真若非空则返回假。

Ps:由于map中存储的是键值对,迭代器为ite,则ite->first为key,ite->second为值

【代码一】

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
#include <iostream>
#include <map>
#include <string>
 
using namespace std;
 
void map_insert(map<string,string> *mapStudent, string index, string x){
    mapStudent->insert(map<string,string>::value_type(index,x));
}
 
int main(){
    char tmp[32]="";
    map<string, string> mapS;
    map_insert(&mapS,"192.168.0.128","xiong");
    map_insert(&mapS,"192.168.200.3","feng");
    map_insert(&mapS,"192.168.200.33","xiongfeng");
     
    map<string, string>::iterator iter;
     
    cout<<"we have there elements"<<endl;
    cout<<"----------------------"<<endl;
     
     
    iter=mapS.find("192.168.200.33");
     
    //查找
    cout<<"查找"<<endl;
    if(iter!=mapS.end()){
        cout<<"find the element"<<endl;
        cout<<"it is:"<<iter->second<<endl;       
    }else{
        cout<<"not find the element"<<endl;
    }
     
    //遍历
    cout<<"遍历"<<endl;
    for(iter=mapS.begin();iter!=mapS.end();iter++){
        cout<<"| "<<iter->first<<" | "<<iter->second<<" |"<<endl;
    }
     
    //删除
    cout<<"删除"<<endl; 
    iter=mapS.find("192.168.200.33");
    if(iter!=mapS.end()){
        cout<<"find the element"<<endl;
        cout<<"delete the element"<<endl;
        mapS.erase(iter);
    }else{     
        cout<<"not find the element"<<endl;
    }
     
    for(iter=mapS.begin();iter!=mapS.end();iter++){
        cout<<"| "<<iter->first<<" | "<<iter->second<<" |"<<endl;
    }
    return 0;
}

 

【代码二】

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
#include <iostream>
#include <map>
#include <cstring>
#include <stdlib.h>
#include <fstream>
 
using namespace std;
 
class PhoneBook
{
public:
    PhoneBook() {}
    ~PhoneBook() {}
    int printall();
    int input(string name,string number);
    string look(string name);
    void exitt();
    int readfile(const string filename);
    int writefile(const string filename);
    int cnt();
 
private:
    map<string,string> phone;
};
 
int PhoneBook::printall()
{
    map<string,string>::iterator ite = phone.begin();
    while(ite!=phone.end()) {
        cout<<ite->first<<"\t";
        cout<<ite->second<<endl;
        ite++;
    }
    return 0;
}
 
string PhoneBook::look(string name){
    map<string,string>::iterator ite = phone.find(name);
    if(ite == phone.end()){
        return "No this user";
    }else{
        return ite->first+"\t\t"+ite->second;
    }
}
 
void PhoneBook::exitt()
{
    exit(0);
}
 
int PhoneBook::readfile(const string filename)
{
    string a,b,c;
    fstream fd;
    fd.open(filename.c_str(),ios::in|ios::binary);
    if(!fd) {
        cout<<"error:write file!"<<endl;
        exit(1);
    }
    phone.clear();
    while(fd>>a>>b>>c){
        phone.insert(pair<string,string>(a,b));
    }
    fd.close();
    return 0;
 
}
 
int PhoneBook::writefile(const string filename)
{
    fstream fd(filename.c_str(),ios::out|ios::binary);
    map<string,string>::iterator ite = phone.begin();
    while(ite!=phone.end()){
        fd<<ite->first<<" "<<ite->second<<" | ";
        ite++;
    }
    fd.close();
    return 0;
}
 
int PhoneBook::input(string name,string number)
{
    //phone[name]=number;
    phone.insert(pair<string,string>(name,number));
    return 0;
}
 
int PhoneBook::cnt(){
    return phone.size();
}
 
 
int main()
{
    PhoneBook *book = new PhoneBook();
    char command[40];
 
    while(true){
        cin>>command;
        if(strcmp(command,"insert")==0){
            string name,number;
            cin>>name>>number;
            book->input(name,number);
        }else if(strcmp(command,"print")==0){
            book->printall();
        }else if(strcmp(command,"search")==0){
            string name;
            cin>>name;
            cout<<book->look(name)<<endl;
        }else if(strcmp(command,"load")==0){
            book->readfile("d:\\12.txt");
        }else if(strcmp(command,"upload")==0){
            book->writefile("d:\\12.txt");
        }else if(strcmp(command,"exit")==0){
            book->exitt();
        }else if(strcmp(command,"count")==0){
            cout<<book->cnt()<<endl;
        }else{
            cout<<"Command Error!\n";
            continue;
        }
    }
    return 0;
}

 

posted @   GGBeng  阅读(178)  评论(0编辑  收藏  举报
编辑推荐:
· 开发中对象命名的一点思考
· .NET Core内存结构体系(Windows环境)底层原理浅谈
· C# 深度学习:对抗生成网络(GAN)训练头像生成模型
· .NET 适配 HarmonyOS 进展
· .NET 进程 stackoverflow异常后,还可以接收 TCP 连接请求吗?
阅读排行:
· 本地部署 DeepSeek:小白也能轻松搞定!
· 基于DeepSeek R1 满血版大模型的个人知识库,回答都源自对你专属文件的深度学习。
· 在缓慢中沉淀,在挑战中重生!2024个人总结!
· 大人,时代变了! 赶快把自有业务的本地AI“模型”训练起来!
· Tinyfox 简易教程-1:Hello World!
点击右上角即可分享
微信分享提示