[数据结构]Map和Set

说起map和set,想必我们都学过红黑树了吧,map和set就是红黑树的一个应用领域。它的底层就是由红黑树来实现的。下面简单说一下map和set的使用吧。

首先,有一个栗子是这样的,让我们统计出每种水果出现的次数。

我们会想到怎么解决的。关于map,我们知道,当你插入同样的key值时,它就不会将要插入的key值插入到map中。但是,我们还知道,map是有俩个参数的,一个是插入的key值,另一个value可以用来统计key值出现的次数。

关于统计水果次数的问题,我们主要有以下几种方法:

    map<string,int> countMap;string strs[] = {"苹果","香蕉","橘子","苹果","苹果","香蕉"};//1.从头到尾遍历看是否有数组中的值,没有则插入,有则使second++for(int i = 0; i < sizeof(strs)/sizeof(strs[0]); i++){map<string,int>::iterator it = countMap.find(strs[i]);if(it != countMap.end()){it->second++;}else{countMap.insert(make_pair(strs[i],1));}}map<string,int>::iterator it1 = countMap.begin();while(it1 != countMap.end()){cout<<it1->first<<":"<<it1->second<<endl;++it1;}
//2.检测插入的返回值pair<map<string,int>::iterator,bool>的second是否为false来确定pair<map<string,int>::iterator,bool> ret;for(int i = 0; i < sizeof(strs)/sizeof(strs[0]);i++){ret = countMap.insert(make_pair(strs[i],1));if(ret.second == false)ret.first->second++;}map<string,int>::iterator it1 = countMap.begin();while(it1 != countMap.end()){cout<<it1->first<<":"<<it1->second<<endl;++it1;}
//3.直接使用map重载的[]for(int i = 0; i < sizeof(strs)/sizeof(strs[0]);i++){countMap[strs[i]]++;}map<string,int>::iterator it1 = countMap.begin();while(it1 != countMap.end()){cout<<it1->first<<":"<<it1->second<<endl;++it1;}

运行结果:

这里写图片描述

这里需要注意的几点是:

1.首先使用插入的时候,是需要插入一个pair结构体,因为map底层的value是一个pair结构体,里边成员又有first和second;因此需要使用make_pair来构造insert的参数。
2.在使用第三种方法的时候,我们使用到map重载的operator[],说一下operator[]的函数:
(*((this->insert(make_pair(x,T()))).first)).second
分解一下上边的operator[]的式子:
this->insert(make_pair(x,T())):返回值为pair<iterator,bool>结构体
((this->insert(make_pair(x,T()))).first):表示pair结构体中的first,即指向一个pair结构体的迭代器,此pair结构体中有key和value,也即所谓的first和second
(*((this->insert(make_pair(x,T()))).first)).second:表示取迭代器中指向的pair的second

说到map,我们还有一个multimap,是用来插入冗余的值,比如有相同的key值的时候,对于map而言,它就不会将其插入,而对于multimap而言就会插入。典型的例子为字典,我们英译汉的时候,同一个英语单词代表着不同的意思,这时multimap就会将每一个key值对应的不同的value值都会插入,并且以排好序的方式显示。

1.比如map来显示字典的时候:

typedef map<string,string> Dict;typedef map<string,string>::iterator DictIt;Dict dict;dict.insert(make_pair("left","左边"));dict.insert(make_pair("right","右边"));dict.insert(make_pair("left","剩余"));DictIt it = dict.begin();while (it != dict.end()){cout<<it->first<<":"<<it->second<<endl;++it;}

结果为:

这里写图片描述

2.用multimap来实现字典的时候:

typedef multimap<string,string> Dict;typedef multimap<string,string>::iterator DictIt;Dict dict;dict.insert(make_pair("left","左边"));dict.insert(make_pair("right","右边"));dict.insert(make_pair("left","剩余"));DictIt it = dict.begin();while (it != dict.end()){cout<<it->first<<":"<<it->second<<endl;++it;}

运行结果:

这里写图片描述

综上所述,
(1)map和multimap可以通过key来找value,也可通过key排序
(2)当我们查找某个key值的时候,发现有多个相同的key值,此时不知道it该指向哪个pair结构体,这里要说明的是,它将返回中序遍历的第一个key值

验证一下:

typedef multimap<string,string> Dict;typedef multimap<string,string>::iterator DictIt;Dict dict;dict.insert(make_pair("left","左边"));dict.insert(make_pair("right","右边"));dict.insert(make_pair("left","剩余"));DictIt it = dict.begin();it = dict.find("left");dict.erase(it);it = dict.begin();while (it != dict.end()){cout<<it->first<<":"<<it->second<<endl;++it;}

在这里,我们找到一个key值为left的pair,此时删除它后再打印一下发现得出的是比它大的value值。

这里写图片描述

说完map和multimap后,与之对应的还有set和multiset。set和multiset是用来判断这个值存在或者不存在。其次也可以用来排序。还有一个特点是过滤(去重)。

1.检测其存在或者不存在

void TestSet()
{typedef set<string> MySet;typedef set<string>::iterator MySetIt;MySet myset;string strs[] = {"苹果","香蕉","橘子","西瓜","草莓","樱桃"};for(int i = 0; i < sizeof(strs)/sizeof(strs[0]); i++){myset.insert(strs[i]);}MySetIt it = myset.begin();if(it == myset.find("哈密瓜"))cout<<"哈密瓜存在"<<endl;elsecout<<"哈密瓜不存在"<<endl;cout<<"存在的其他水果为:"<<endl;it = myset.begin();while(it != myset.end()){cout<<*it<<endl;++it;}
}

运行结果:

这里写图片描述

2.排序和去重

typedef set<int> MySet;typedef set<int>::iterator MySetIt;MySet myset;for(int i = 10; i > 0; i--)myset.insert(i);myset.insert(5);MySetIt it = myset.begin();while(it != myset.end()){cout<<*it<<endl;++it;}

运行结果:

这里写图片描述

对于过滤来说,就是如果有相同的key值,它就会去掉相同的key,只插入一个到set中。

这里multiset的意义和multimap差不多,也是处理冗余的数据。使用方法类似。

对于map和set的底层是怎么实现的呢。它是通过写的一个红黑树。主要的区别是

里边的value_type的意义,对于map来说,value_type指的是一个pair的结构体,结构体成员为key和value,而对于set来说,value_type指的是key值。
在红黑树中,用了枚举来表示颜色。而在源码的红黑树中使用了bool值来代替红黑俩种颜色
我们还知道,map和multimap,set和multiset也有区别,底层是怎么用红黑树的呢。它是插入的时候分别对红黑树的插入分为唯一插入(insert_unique)和相等插入(insert_equal)(相等插入就是对冗余数据的考虑)。
set的插入:

map中value_type: typedef pair<const Key, T> value_type;
set中value_type: typedef Key value_type;

set的插入:

  typedef  pair<iterator, bool> pair_iterator_bool; pair<iterator,bool> insert(const value_type& x) { pair<typename rep_type::iterator, bool> p = t.insert_unique(x); return pair<iterator, bool>(p.first, p.second);}

map的插入:

pair<iterator,bool> insert(const value_type& x) 
{ return t.insert_unique(x); }

multimap的插入:

  iterator insert(const value_type& x) { return t.insert_equal(x); }

multiset的插入:

  iterator insert(const value_type& x) { return t.insert_equal(x);}

还可以进行相关的区间的插入,删除,某个位置的插入删除等操作。

小姿势:

lower_bound : 用来找到比key值大的数
upper_bound : 用来找到比key值大的数

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/382712.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

js获取Json对象的长度

有两种Json形式&#xff1a; 第一种&#xff1a; var json1 {"data":[{"name":"zs","age":"10"}]};对于这种格式的json数据&#xff0c;如果想获取data的长度&#xff0c;就可以用以下这种方式&#xff1a; var length …

生产者消费者模型(条件变量)

三种关系&#xff1a;互斥&#xff0c;同步&#xff0c;互斥和同步 两类角色:生产者&#xff0c;消费者&#xff08;线程&#xff09; 一个交易场所&#xff1a;生产者消费者共享的区域 卖苹果的模型 dish上面只有一个苹果买家必须要等卖家把苹果放到dish上才可以去买苹果。…

linux之信号

信号&#xff1a;在生活中&#xff0c;我们遇到过不同种类的信号&#xff0c;比如&#xff1a;&#xff08;交通信号&#xff0c;乃至某个人的表情&#xff0c;动作等带给你不同的信号&#xff09;然而&#xff0c;在我们的linux下&#xff0c;我们最熟悉的就是&#xff0c;当遇…

视频解析有感,在解析 iqiyi与qq视频的时候,记录一些发现

最近对iqiyi与qq视频解析发现&#xff0c;两个网站的解析流程&#xff0c;尤其是反解析措施 各有特点&#xff0c;简单记录一下 先说iqiyi&#xff0c; 浏览器模拟移动端可以拿到视频的mp4链接&#xff0c;这个不多说。 iqiyiPC端浏览器获取 ts过程&#xff1a; a.iqiyi一次性…

C语言atoi函数的用法

#include < stdlib.h > int atoi(const char *nptr);用法&#xff1a;将字符串里的数字字符转化为整形数。返回整形值。 注意&#xff1a;转化时跳过前面的空格字符&#xff0c;直到遇上数字或正负符号才开始做转换&#xff0c;而再遇到非数字或字符串结束时(’/0’)才…

[Linux]继续探究mysleep函数(竞态条件)

之前我们探究过mysleep的简单用法&#xff0c;我们实现的代码是这样的&#xff1a; #include<stdio.h> #include<signal.h>void myhandler(int sig) {}unsigned int mysleep(unsigned int timeout) {struct sigaction act,oact;act.sa_handler myhandler;sigempt…

C语言的atoi和C++的to_string

to_stringint to string将其他型转换成字符串型atoiascii to integer是把字符串转换成整型数的一个函数 to_string #include <iostream> // std::cout #include <string> // std::string, std::to_stringint main () {std::string perfect std::to_string…

ubuntu 升级python3.5到python3.7,并升级pip3

1, 下载python3.7.tgz 文件&#xff0c;解压&#xff0c; 2. 编译安装 3. 删除 /usr/bin 目录下的 pip3, python3 4. 建立新的软连接&#xff1a; #添加python3的软链接ln -s /usr/local/python3/bin/python3.7 /usr/bin/python3#添加 pip3 的软链接ln -s /usr/local/python3/b…

[Linux]死锁

死锁是指多个进程在运行过程中因争夺资源而造成的一种僵局&#xff0c;当进程处于这种僵持状态时&#xff0c;若无外力作用&#xff0c;它们都将无法再向前推进。之前信号量的时候我们知道&#xff0c;如果多个进程等待&#xff0c;主要体现在占有锁的问题上。死锁也可以被定义…

Python安装第三方模块总结 转载的

转自 https://www.jellythink.com/archives/541

[C++]vector创建二维数组

c.resize(n);将c重置为大小为n个元素向量&#xff0c;如果n比原来的元素多&#xff0c;则多出的元素常被初始化为0//节选《面向对象的程序设计》杜茂青 int N5, M6; vector<vector<int> > Matrix(N); for(int i 0; i< Matrix.size(); i){ Matrix[i].resize(M…

[Linux]线程安全和可重入函数

线程安全&#xff1a;一个函数被称为线程安全的&#xff0c;当且仅当被多个并发进程反复调用时&#xff0c;它会一直产生正确的结果。如果一个函数不是线程安全的&#xff0c;我们就说它是线程不安全的。 重入&#xff1a;函数被不同的控制流程调用,有可能在第一次调用还没返回…

[Linux]信号量

信号量是一个计数器&#xff0c;用于为多个进程提供对共享数据对象的访问。 在信号量上只有三种操作可以进行&#xff0c;初始化、递增和增加&#xff0c;这三种操作都是原子操作。递减操作可以用于阻塞一个进程&#xff0c;增加操作用于解除阻塞一个进程。 为了获得共享资源…

Linux VIM 程序中有游离的‘\357’ ‘\274’错误

gcc date.cpp -o date -lstdc date.cpp:18:20: 错误&#xff1a;程序中有游离的‘\357’date.Showdata()&#xfffd;&#xfffd;&#xfffd;^ date.cpp:18:21: 错误&#xff1a;程序中有游离的‘\274’date.Showdata()&#xfffd;&#xfffd;&#xfffd;^ date.cpp:18:22…

[Linux]关于SIGCHLD

之前我们就学过&#xff0c;关于wait和waitpid来处理僵尸进程&#xff0c;父进程等待子进程结束后自己才退出&#xff0c;这样的方法有俩种方式&#xff0c;一种是父进程死死的等子进程退出&#xff0c;也就是使用阻塞的方式等待子进程退出&#xff0c;另一种方式是通过非阻塞的…

C语言思维导图

本人能力有限&#xff0c;知识点难免概括不全&#xff0c;如有错误欢迎指正

转载一篇关于curl的文章

转载一篇关于curl的文章 http://www.360doc.com/content/16/0107/15/18578054_526158476.shtml

[Linux]vi/vim下添加多行注释和取消注释

添加注释&#xff08;Centos&#xff09;&#xff1a; 在命令行模式下按ctrlV进入 visual block模式&#xff08;可视化模式&#xff09; 选中你需要注释的行&#xff0c;再按大写的I&#xff0c;输入//&#xff0c;最后按俩下esc即可。 如果想让前进tab个位&#xff0c;则可在…

pthread和互斥量条件变量函数意义速查表

数据类型 pthread_t 线程 互斥量和条件变量

[Linux]共享内存

共享内存是UNIX提供的进程间通信手段中速度最快的一种&#xff0c;也是最快的IPC形式。为什么是最快的呢&#xff0c;因为数据不需要在客户进程和服务器进程之间复制&#xff0c;所以是最快的一种IPC。这是虚存中由多个进程共享的一个公共内存块。 两个不同进程A、B共享内存的…