A Brief Introduction to Multiset[STL]
基础
multiset是<set>库中一个非常有用的类型,它可以看成一个序列,插入一个数,删除一个数都能够在O(logn)的时间内完成,而且他能时刻保证序列中的数是有序的,而且序列中可以存在重复的数。
我们通过一个程序来看如何使用multiset。
#include <string>
#include <iostream>
#include <set>
using namespace std;
int main(){
int x;
scanf("%d",&x);
multiset<int>h;//初始h为空
while(x!=0){
h.insert(x);//将x插入h中
scanf("%d",&x);
}
while(!h.empty()){
__typeof(h.begin()) c=h.begin();//c指向h序列中第一个元素的地址,第一个元素是最小的元素
printf("%d ",*c);
h.erase(c);//从h序列中将c指向的元素删除
}
}
对于输入数据32 61 12 2 12 0,该程序的输出是2 12 12 32 61。
当要在h中插入一个数x时,语法为h.insert(x);
当在h中删除指针c指向的元素*c时,语法为h.erase(c)。
注意,如果我们把h.erase(c)写成h.erase(*c),那么该语句就会把h中所有和*c相等的元素都删掉
如果要查找最大的元素并赋值给k,语法是
int k=*(h.end()--);
end 指向 just beyond the last element
如果要想知道当前序列中比k大的元素最小的是多少,那么可以这样
int p=*(h.upper_bound(k));
其中h.upper_bound(k)表示比 k 大的最小的数的地址。
不光是int类型,multiset还可以存储其他的类型诸如 string类型,结构(struct或class)类型。而我们一般在编程当中遇到的问题经常用到多关键字的类型,即struct或class。
例如下面的例子:
struct rec{
int x,y;
};
multiset<rec>h;
以上的代码是没有任何用处的,因为multiset并不知道如何去比较一个自定义的多关键字类型。
我们可以定义multiset里面rec类型变量之间的小于关系的含义(这里以x为第一关键字为例),具体过程如下:
我们定义一个比较类cmp,cmp内部的operator函数的作用是比较rec类型a和b的大小(以x为第一关键字,y为第二关键字):
struct cmp{
bool operator()(const rec&a,const rec&b){
return a.x<b.x||a.x==b.x&&a.y<b.y;
}
};
此时rec以及multiset的定义部分完整代码可参考如下:
struct rec{
int x,y;
};
struct cmp{
bool operator()(const rec&a,const rec&b){
return a.x<b.x||a.x==b.x&&a.y<b.y;
}
};
multiset<rec,cmp>h;
通过以上代码,我们就能建立一个集合h使得该集合能够存储和排序多关键字类型
我们来看一个小应用:求从一个点到另一个点的最短路长度,边都是正权。
正权边的最短路问题可以用dijkstra算法来解决,而优化dijkstra算法可以用heap。这里我们来看如何用multiset实现dijkstra+heap。
以下代码省去了输入输出和图的建立。我们光看求最短路的部分。
y代表图中点的编号,而x则代表当前点y与源点的最短距离。
d[0]=0;//源点是0
rec a;
a.x=0;
a.y=0;
h.insert(a);
while(!h.empty()){//维护已扩展的点距离由小到大
__typeof(h.begin()) c = h.begin();
rec t = (*c);
h.erase(c);
for(int i=tail[t.y];i;i=next[i]){//前向星存边
int j = p[i];
rec a;
if(d[j]==-1){//d[j]==-1表示j还没有被访问
d[j] = t.x+w[i];//扩展
a.x = d[j];
a.y = j;
h.insert(a);//加入堆
}
else if(d[j] > t.x + w[i])
{//松弛操作(因为需要删除,所以是否访问过要分情况处理)
a.x = d[j];
a.y = j;//先构造旧的a, 用来找到a
c = h.upper_bound(a);
c--;
h.erase(c);//删掉旧的a
a.x = t.x + w[i];
d[j] = a.x;
h.insert(a);//插入新的a
}
}
}
有了multiset类型,我们就不用再去写平衡树一类的东西了,从而大大降低了编程复杂度.
方法
begin() //返回指向第一个元素的迭代器
clear() //清除所有元素
count() //返回某个值元素的个数
empty() //如果集合为空,返回true
end() //返回指向最后一个元素之后的迭代器,不是最后一个元素
erase() //删除集合中的元素
find() //返回一个指向被查找到元素的迭代器
insert() //在集合中插入元素
upper_bound() //返回大于某个值元素的迭代器
lower_bound() //返回指向大于(或等于)某值的第一个元素的迭代器
rbegin() //返回指向集合中最后一个元素的反向迭代器
rend() //返回指向集合中第一个元素的反向迭代器
size() //集合中元素的数目 equal_range() //返回集合中与给定值相等的上下限的两个迭代器
get_allocator() //返回集合的分配器
key_comp() //返回一个用于比较元素键值的函数
value_comp() //返回一个用于比较元素值的函数
max_size() //返回集合能容纳的元素的最大限值
swap() //交换两个集合变量
关于反向迭代器(随后添加超链接)
集合操作
std::set_intersection();
std::set_union() ;std::set_difference();std::set_symmetric_difference();
struct compare{bool operator ()(string s1,string s2){return s1>s2;}///自定义仿函数};
转自百度, 有改动.
A Brief Introduction to Multiset[STL]的更多相关文章
- ZOJ 3963 Heap Partition(multiset + stl自带二分 + 贪心)题解
题意:给你n个数字s1~sn,要你把它们组成一棵棵二叉树,对这棵二叉树来说,所有节点来自S,并且父节点si<=子节点sj,并且i<j,问你树最少几棵二叉数.树 思路:贪心.我们往multi ...
- HDU4022 Bombing STL
Bombing Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65768/65768 K (Java/Others) Total Su ...
- for_each的各种情况下的使用详解
原创作者:http://oomusou.cnblogs.com 配合<C++ Template>(简体中文)使用 http://download.csdn.net/detail/qq239 ...
- HDU 4022 Bombing(stl,map,multiset,iterater遍历)
题目 参考了 1 2 #define _CRT_SECURE_NO_WARNINGS //用的是STL中的map 和 multiset 来做的,代码写起来比较简洁,也比较好容易理解. ...
- STL vector+sort排序和multiset/multimap排序比较
由 www.169it.com 搜集整理 在C++的STL库中,要实现排序可以通过将所有元素保存到vector中,然后通过sort算法来排序,也可以通过multimap实现在插入元素的时候进行排序.在 ...
- STL的基本使用之关联容器:set和multiSet的基本使用
STL的基本使用之关联容器:set和multiSet的基本使用 简介 set 和 multiSet 内部都是使用红黑树来实现,会自动将元素进行排序.两者不同在于set 不允许重复,而multiSet ...
- STL之set && multiset
一.set 在了解关联容器set之前,让我们先来看看下面这个例子,并猜测该例子输出什么: // stl/set1.cpp #include <iostream> #include < ...
- 转自http://blog.sina.com.cn/daylive——C++ STL set&multiset
C++ STL set和multiset的使用 1,set的含义是集合,它是一个有序的容器,里面的元素都是排序好的,支持插入,删除,查找等操作,就 像一个集合一样.所有的操作的都是严格在logn时间 ...
- STL - set和multiset
set/multiset的简介 set是一个集合容器,其中所包含的元素是唯一的,集合中的元素按一定的顺序排列.元素插入过程是按排序规则插入,所以不能指定插入位置. set采用红黑树变体的数据结构实现, ...
随机推荐
- 在windows后台调用webservice
1.首先要创建个webservice,然后再webservice写一个方法如图 2.然后将WebService1.asmx 在浏览器中浏览会出现如图所示(该地址很重要,复制此地址在下边程序中要用到) ...
- Java threadpool机制深入分析
简介 在前面的一篇文章里我对java threadpool的几种基本应用方法做了个总结.Java的线程池针对不同应用的场景,主要有固定长度类型.可变长度类型以及定时执行等几种.针对这几种类型的创建,j ...
- [互联网面试笔试汇总C/C++-9] 实现赋值运算符函数-剑指offer
题目:如下为类型CMyString的声明,请为该类型添加赋值运算符函数. class CMyString { public: CMyString(char* pData = NULL); CMyStr ...
- Linux 查看文件
1: cat file_full_name 这种方法适合查看小文件 cat /etc/services cat -n file_full_name -n 可以加行号! 2: tac file_ful ...
- 转: angular编码风格指南
After reading Google's AngularJS guidelines, I felt they were a little too incomplete and also guide ...
- Codeforces 706D Vasiliy's Multiset(可持久化字典树)
[题目链接] http://codeforces.com/problemset/problem/706/D [题目大意] 要求实现一个集合中的三个操作,1:在集合中加入一个元素x,2:从集合中删除一个 ...
- POJ 3581 Sequence(后缀数组)
[题目链接] http://poj.org/problem?id=3581 [题目大意] 给出一个数列,将这个数列分成三段,每段分别翻转,使得其字典序最小,输出翻转后的数列. [题解] 首先,第一个翻 ...
- 《windows程序设计》学习_3.3:利用xp扫雷资源
#include<windows.h> #include "resource.h" LRESULT CALLBACK WndProc (HWND, UINT, WPAR ...
- android之获取应用中的图片资源_获取找你妹中的图片资源
一直不知道原来获取一个应用中的图片资源这么简单,刚才直接把apk解压,就得到了里面的一下文件,搜索一下就全部把图片资源找出来了,想要模仿应用或者自己不会ui的话,用现成的资源方便多了. 也没多少说的, ...
- Fastboot的使用简单教程
大家都知道HTC手机重新启动进入所谓的project模式,就是HBOOT,然后能够进入FASTBOOT界面,在这个界面.我们能够在电脑端能够做非常多事,特别是HBOOT被改动过,假设是ENG S-OF ...