C++语言中multiset的相关用法及扩展

***************************************************

更多精彩,欢迎进入:http://shop115376623.taobao.com

***************************************************


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

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

#include <string>
#include <iostream>
#include <set>
using namespace std;
void main()

{
     int x;
     scanf("%ld",&x);
     multiset<inth;//建立一个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类型,我们就不用再去写平衡树一类的东西了,从而大大降低了编程复杂度

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

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

相关文章

[20160201]db_link与子光标问题.txt

[20160201]db_link与子光标问题.txt --生产系统遇到一个关于db_link产生大量子光标问题&#xff0c;当cursor_sharingforce的情况下&#xff0c;通过测试说明。 --注&#xff1a;这个问题我的测试仅仅存在10.2.0.4,11.2.0.4没有这个问题。 1.环境&#xff1a; SCOTTtest> &a…

脚本命令配置mysql_MySQL 自动化部署脚本

一、环境说明操作系统&#xff1a;CentOS数据库版本&#xff1a;MySQL 5.7/8.0参数&#xff1a;buffer pool 会根据系统内存指定、默认双一、GTID、SlowLog脚本默认安装路径&#xff1a;/usr/local/mysql脚本默认数据路径&#xff1a;/data/mysql*(根据安装包版本适应 比如 5.7…

STL 整理(map、set、vector、list、stack、queue、deque、priority_queue)

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com STL视频教程&#xff1a; http://item.taobao.com/item.htm?spma1z10.5-c.w4002-9510581626.21.y9vLuz&id43055362725 ************…

第2章 数字之魅——快速寻找满足条件的两个数

快速寻找满足条件的两个数 问题描述 能否快速找出一个数组中的两个数字&#xff0c;让这两个数字之和等于一个给定的数字&#xff0c;为了简化起见&#xff0c;我们假设这个数组中肯定存在这样一组或以上符合要求的解。 分析与解法 【解法一】 代码如下&#xff1a; 1 package …

eigen 列向量转矩阵_快速入门矩阵运算——开源库Eigen

矩阵是数学中一个重要的工具&#xff0c;广泛应用于各种场景下的数值分析&#xff0c;例如&#xff0c;数字信号处理&#xff0c;图像处理等。我们如何在程序中使用矩阵进行运算呢&#xff1f;本文将为大家介绍一个开源的矩阵运算工具——Eigen。Eigen is a C template library…

优化android studio编译的apk大小

默认生成的apk&#xff08;release版&#xff09;太大&#xff08;只打印helloworld的apk大小就有1MB多&#xff09;&#xff0c;需要优化编译&#xff0c;如下&#xff0c;修改build.gradlebuildTypes { release {minifyEnabled true zipAlignEnabled true s…

STL set和multiset

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com STL视频教程&#xff1a; http://item.taobao.com/item.htm?spma1z10.5-c.w4002-9510581626.21.y9vLuz&id43055362725 ***************…

cassandra 环境搭建

1 下载安装包 http://www.planetcassandra.org/cassandra/?dlinkhttp://downloads.datastax.com/community/dsc-cassandra-2.1.5-bin.tar.gz 用tar包的形式安装&#xff0c;解压缩之后就可以用&#xff1b;选择2.1.5&#xff0c;因为golang 客户端gocql的git 首页上支持最新的…

mysql raid_DBA们应该知道的RAID卡知识_MySQL

bitsCN.com对于数据库这种特殊应用IOphotoshop/ target_blank classinfotextkey>PS往往会成为瓶颈&#xff0c;突破的这个瓶颈的有效方法不多&#xff0c;软件方面主要是读写分离&#xff0c;垂直拆分&#xff0c;分区表技术&#xff0c;cluster。硬件方面主要是raid&#x…

基于Maven的SSH框架搭建

2019独角兽企业重金招聘Python工程师标准>>> 1.工程介绍 工程是结合了Springstruts2hibernate&#xff0c;实现了一个简单的form表单提交的功能&#xff0c;可能需要对spring&#xff0c;struts2&#xff0c;hibernate有一个基础的了解才好理解。 2.工程结构图 首先…

交通警察手势信号(动画演示)

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com *************************************************** 一、交通警察手势信号&#xff0d;停止信号 二、交通警察手势信号&#xff0d;直行信…

==与equal在java中应用的感悟

今天又算是长见识了。了解了下平时不注意的equal和的区别。 不管是又或是equal都是用来比较相同与否。当问题就在这里了&#xff0c;比较什么相同呢&#xff1f; 我的在日常的比较无非也就是两种&#xff1a;1、基本数据类型之间的比较&#xff0c;2、引用类型数据之间的比较。…

mysql和mybatis面试题_BATJ面试题汇总详解:MyBatis+MySQL+Spring+Redis+多线程

SpringSpring 概述什么是spring?使用Spring框架的好处是什么&#xff1f;Spring由哪些模块组成&#xff1f;解释AOP模块Spring配置文件什么是Spring IOC 容器&#xff1f;依赖注入什么是Spring的依赖注入&#xff1f;有哪些不同类型的IOC(依赖注入)方式&#xff1f;哪种依赖注…

Codeblocks和gdb调试 (转)

*************************************************** 更多精彩&#xff0c;欢迎进入&#xff1a;http://shop115376623.taobao.com *************************************************** 使用C::B和gdb调试是一件简单的事情。下面&#xff0c;让我们调试一个简单的循环&…

Git的安装和使用(Linux)【转】

转自&#xff1a;http://my.oschina.net/fhd/blog/354685 Git诞生于Linux平台并作为版本控制系统率先服务于Linux内核&#xff0c;因此在Linux上安装Git是非常方便的。可以通过两种不同的方式在Linux上安装Git&#xff1a;一种方法是通过Linux发行版的包管理器安装已经编译好的…

Magento--修改已存在的订单的运费

遇到一种情况&#xff0c;需要在下单后再由管理员添加订单运费&#xff0c;然后顾客再付款。那么问题来了&#xff0c;如何给订单添加运费呢&#xff1f;下面是一段代码&#xff0c;可以实现该功能&#xff1a; $orderId your order id;$order Mage::getModel(sales/order)-&…

mysql5.7.17 win7_win7下mysql5.7.17安装配置方法图文教程

win7下安装mysql5.7.17图文教程&#xff0c;分享给大家。1.下载安装包请在以下有zip包和msi两种类型包&#xff0c;建议新手选择zip包下载&#xff0c;有助于熟习mysql2.解压mysql压缩包下载完成后解压&#xff0c;将其放在要安装的目录下面&#xff0c;如&#xff1a;e:\mysql…

华为软件研发面试题1

from&#xff1a;http://blog.csdn.net/eddy_0825/article/details/1875574 1、局部变量能否和全局变量重名&#xff1f; 答&#xff1a;能&#xff0c;局部会屏蔽全局。要用全局变量&#xff0c;需要使用"::" 局部变量可以与全局变量同名&#xff0c;在函数内引用这…

停一下

15年过去已半载有余&#xff0c;回头看年初定下的目标&#xff0c;有种管中窥豹的感觉。之前和肉山讨论的时候&#xff0c;他对我想要发展的方向并没有表示赞同。 现在认为他是对的&#xff0c;发展的方向太靠前了&#xff0c;ui&#xff0c;canvas&#xff0c;svg&#xff0c;…

which 命令

我们经常在linux要查找某个文件&#xff0c;但不知道放在哪里了&#xff0c;可以使用下面的一些命令来搜索&#xff1a; which 查看可执行文件的位置。 whereis 查看文件的位置。 locate 配合数据库查看文件位置。 find 实际搜寻硬盘查询文件名…