[ZZ] C++ Multiset

cpp语言中,multiset是<set>库中一个非常有用的类型,它可以看成一个序列,插入一个数,删除一个数都能够在O(logn)的时间内完成,而且他能时刻保证序列中的数是有序的,而且序列中可以存在重复的数。

我们通过一个程序来看如何使用multiset。

#include <string>
#include <iostream>
#include <set>
using namespace std;
void main(){
    intx;
    scanf("%ld",&x);
    multiset<int>h;//建立一个multiset类型,变量名是h,h序列里面存的是int类型,初始h为空
    while(x!=0){
        h.insert(x);//将x插入h中
        scanf("%ld",&x);
    }    
    while(!h.empty()){// 序列非空 h.empty()==true时 表示h已经空了
        __typeof(h.begin()) c=h.begin();//c指向h序列中第一个元素的地址,第一个元素是最小的元素
        printf("%ld ",*c);//将地址c存的数据输出
        h.erase(c);//从h序列中将c指向的元素删除
    }
}
对于输入数据32 61 12 2 12 0,该程序的输出是2 12 12 32 61。

我们可以看到,当一个变量h被定义为multiset类型时。所有关于它的操作可以写成如下格式:

h.函数名(形参);

当要在h中插入一个数x时,语法为h.insert(x);当在h中删除指针c指向的元素*c时,语法为h.erase(c)。

注意,如果我们把h.erase(c)写成h.erase(*c),那么该语句就会把h中所有和*c相等的元素都删掉,大家要注意

如果要查找最大的元素并赋值给k,语法是int k=*(h.end()--),注意multiset类型的尾地址存的内容是空的。

如果要想知道当前序列中比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;
    }
};

然后我们将语句"multiset<rec>h;” 改成"multiset<rec,cmp>h;"这样以后,我们就告诉了序列h如何去比较里面的元素(这种方法属于重载运算符,在编程当中经常用到,这里就不详细介绍了)

此时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。

以下代码省去了输入输出和图的建立。我们光看求最短路的部分。注意,这里的多关键字类型名称还是rec,multiset集合的名称还是h;多关键字类型rec中,x是第一关键字,y是第二关键字,y代表图中点的编号,而x则代表当前点y与源点的最短距离。

d[0]=0;//源点是0
reca;
a.x=0;//第一关键字x表示距离
a.y=0;//第二关键字y表示点的编号
h.insert(a);//将a插入序列中
while(!h.empty()){//h集合中的元素是否为空
      __typeof(h.begin()) c=h.begin();
    rect=(*c);//取最小值
    h.erase(c);//将最小值删去
    for(inti=tail[t.y];i;i=next[i]){
        intj=p[i];//枚举和t.y相邻的点
        reca;//建立一个结构类变量a
        if(d[j]==-1){//d[j]==-1表示j还没有被访问
            d[j]=t.x+w[i];//w[i]表示边i的边权
            a.x=d[j];
            a.y=j;//将j的相关信息保存在rec类型a中
            h.insert(a);
        }elseif(d[j]>t.x+w[i]){//最短路算法的松弛操作
            a.x=d[j];
            a.y=j;//将j在序列中的信息存储到a中
            c=h.upper_bound(a);//找到序列h中a之后的元素的地址
            c--;//地址减一就是a所在的地址
            h.erase(c);//删掉a
            a.x=t.x+w[i];
            d[j]=a.x;//更新最短路的值
            h.insert(a);//插入
        }
    }
}
有了multiset类型,我们就不用再去写平衡树一类的东西了,从而大大降低了编程复杂度

时间: 2024-07-28 21:19:20

[ZZ] C++ Multiset的相关文章

zoj Median (multiset)

Median Time Limit: 5 Seconds      Memory Limit: 65536 KB The median of m numbers is after sorting them in order, the middle one number of them ifm is even or the average number of the middle 2 numbers if m is odd. You have an empty number list at fir

没为类型 Node 定义方法 getTextContent (zz)

没有为类型 Node 定义方法 getTextContent (zz) 晚上下班的时候,把班上写了半截的代码带了回来.结果回到家后出乎意料的是回来的时候将代码导入eclipse后,下面这行代码就直接报错了,显示 getTextContent()未定义 . ((Element) ele.getElementsByTagName( "err_code").item(0 )).getTextContent(); 首先想到的是jdk 的版本问题,不可能啊,我昨天才装的jdk 1.6.0_24

hdu 4268 Alice and Bob(multiset)

# include<stdio.h> # include<algorithm> # include<iostream> # include<string.h> # include<map> # include<set> # include<vector> using namespace std; struct node { int h; int w; }; struct node a[100010],b[100010];

Linux 信号signal处理机制(ZZ)

http://www.cnblogs.com/taobataoma/archive/2007/08/30/875743.html 信号是Linux编程中非常重要的部分,本文将详细介绍信号机制的基本概念.Linux对信号机制的大致实现方法.如何使用信号,以及有关信号的几个系统调用. 信号机制是进程之间相互传递消息的一种方法,信号全称为软中断信号,也有人称作软中断.从它的命名可以看出,它的实质和使用很象中断.所以,信号可以说是进程控制的一部分. 一.信号的基本概念 本节先介绍信号的一些基本概念,然后

第十三篇:multimap容器和multiset容器中的find操作

前言 multimap容器是map容器的“ 增强版 ”,它允许一个键对应多个值.对于map容器来说,find函数将会返回第一个键值匹配元素所在处的迭代器.那么对于multimap容器来说,find函数将如何运作呢?如果要实现和map容器的find函数同样的功能,则它将返回多个迭代器,这样太复杂了.本文将讲解C++中multimap容器的“ find实现 ”. 解决思路一 摒弃find函数,使用另外两个新函数,它们是专家们为了解决multimap中的“ find操作 ”问题专门设计的: 1. lo

hdu 5578 Friendship of Frog(multiset的应用)

Problem Description N frogs from different countries are standing in a line. Each country is represented by a lowercase letter. The distance between adjacent frogs (e.g. the 1st and the2nd frog, the N−1th and the Nth frog, etc) are exactly 1. Two fro

【multiset】hdu 5349 MZL&#39;s simple problem

[multiset]hdu 5349 MZL's simple problem 题目链接:hdu 5349 MZL's simple problem 题目大意 n次操作,插入元素.删除最小元素.查询最大元素并输出. C++STL的multiset的使用 set--多元集合(元素不可重复),multiset--可重复元素的多元集合 多元集合(MultiSets)和集合(Sets)相像,只不过支持重复对象.(具体用法请参照set容器) set和multiset内部是以平衡二叉树实现的: 从内部数据结

实例讲解,set,multiset,map,multimap关联容器

测试环境:windows 7 vs2010 内部元素有序排列,新元素插入的位置取决于它的值,查找速度快. 除了各容器都有的函数外,还支持以下成员函数: find: 查找等于某个值的元素(x小于y和y小于x同时不成立即为相等) lower_bound: 查找某个下界 upper_bound: 查找某个上界 equal_range: 同时查找上界和下界 count:计算等于某个值的元素个数(x小于y和y小于x同时不成立即为相等) insert: 用以插入一个元素或一个区间 在学习关联容器之前,我们先

HDU 4268 Alice and Bob(贪心+multiset)

HDU 4268 题意:Alice与Bob在玩卡片游戏,他们每人有n张卡片,若Alice的一张卡片长与宽都不小于Bob的一张卡片,则Bob的卡片就会被盖住,一张卡片只可以使用一次,且不可旋转求Alice最多可以盖住多少张Bob的卡片. 思路:记录两人卡片情况,并按照长度将两人卡片分别降序排序.遍历两人的卡片,将长度小于Alice的卡片长度的Bob卡片的宽度插入multiset中,在multiset中找到小于等于Alice卡片宽度的第一个数,将这个数给消去且答案+1.//贪心法自行发挥即可. co