数据结构——优先级队列及多服务台模拟系统的实现

news2025/1/10 11:48:44

一、优先级队列的定义和存储

优先级队列定义:优先级高的元素在队头,优先级低的元素在队尾

基于普通线性表实现优先级队列,入队和出队中必有一个时间复杂度O(n),基于二叉树结构实现优先级队列,能够让入队和出队时间复杂度都为O(logn),这种基于树状结构的优先级队列也称为二叉堆。当根结点为最大元素时,称为最大化堆或大顶堆;当根结点为最小元素时,称为最小化堆或小顶堆。二叉堆是有序的完全二叉树,使用顺序存储,留出数组第一个位置作为哑结点,如图所示:

二、优先级队列的运算实现

先学习优先级队列的两大核心运算:入队和出队,之后再介绍如何建堆。下面以小顶堆为例分析。

2.1 入队:上滤插入

在原有的二叉堆上插入新元素,同时使其维持有序性,可以通过“上滤”实现。第一步是寻找新元素应插入的位置,第二步是插入新元素,时间复杂度为O(logn)

//先将新元素放在二叉堆最底部
int hole=++currentlength;
//与父结点比较,决定是否交换(实际上不需要交换,因为新元素不一定到达最终插入位置),条件:比父结点值小,上滤过程持续进行,设置循环;
//循环结束条件:到达堆顶(hole=1)或已到达最终插入位置
while(hole>1&&x<array[hole/2])
{
	array[hole]=array[hole/2]; //父结点下移
	hole/=2; //更新新元素位置
}
//插入元素
array[hole]=x;

 2.2 出队:下滤删除

现在要删除队头元素,也就是堆顶元素,简单的想法是和孩子节点比较,选择其中较小的上滤,可这样可能会破坏二叉堆完全二叉树的结构[比如1 3 2 4 5删除后2 3 null 45],为了维持完全二叉树的结构,我们需要考虑为堆最后一个结点重新寻找位置。如图所示,将最后一个节点放在堆顶再下滤,就能实现删除堆顶元素并使二叉树结点数量减一。[比如1 3 2 4 5->

5 3 2 4->2 3 5 4],时间复杂度为O(logn)

//把最后一个元素放在堆顶
tmp=array[1]=array[currentlength--];
hole=1;
//叶子结点条件 2i>n,非叶子结点条件:2i<=n
while(2*hole<=currentlength){
	//选择较小子节点
	child=2*hole;
	if(2*hole!=currentlength) 
	  if(array[child]>array[child+1])  child++;
	//下滤
	if(tmp>array[child]) {
		array[hole]=array[child];
		hole=child;
	}
	//否则下滤结束
	else
	break;	
	  //否则选择左孩子结点上滤
}

2.3 建堆 

 

在构造函数中传入数据数组初始化二叉堆,最简单的想法是对N个元素执行N次入队操作,每次入队后对维持二叉堆有序性,时间复杂度O(nlogn);如何提升时间效率呢,可以先用原始数据初始化二叉堆,再从最后一个非叶节点开始到第一个结点,依次执行下滤操作,这样调用percolateDown(i)时保证结点i的所有自堆都满足有序性,这样自下往上,从局部有序最后抵达全局有序,可以证明这种建堆方式时间复杂度为O(n)。

对2.2中的代码略微修改容易得到下滤函数:

//下滤函数 precolateDown实现
tmp=array[i];
hole=i;
while(2*hole<=currentlength){
	child=2*hole;
	if(2*hole!=currentlength) 
	  if(array[child]>array[child+1])  child++;

	if(tmp>array[child]) {
		array[hole]=array[child];
		hole=child;
	}
	else
	break;	

建堆过程可以表示为

//建堆 buildHeap 实现
for(int i=currentlength/2;i>0;i--){
    precolateDown(i);
}

三、 STL中的优先级队列

类模板名:priority_queue

头文件:queue

定义:priority<elemtype,base container(默认 vector),(默认大顶堆,添加谓词greater<int>则定义小顶堆)>

成员函数:

void push() 入队

void pop() 出队

Elemtype top() 获取队头元素

void clear() 清空

bool empty() 判空

 四、D堆

D堆就是D叉树,由于生成的堆更矮,入队的时间复杂度为O(log_{D}N),比二叉堆更加优越。

但出队时就不一样了,由于需要比较子结点,若采用最简单的选择算法,则时间复杂度为O(Nlog_{D}N),因此D堆适用于插入操作比删除操作多非常多的场景。

五、归并优先级队列(了解)

归并:即合并两棵有序树,在前面我们做过合并二叉树的题目,归并可通过递归实现

5.1 左堆

这里的nql容易通过递归实现,参考二叉树中刷题中的类似题目。

 

5.2 斜堆 

5.3 二项堆 

二项堆和二进制有很大相似性,归并的过程也像二进制加法

 

六、多服务台排队系统模拟

#include<queue>
#include<ctime>
#include<cstdlib>
#include<iostream>
using namespace std;

class simulator {  //模拟类
private:
	enum IO { IN, OUT };
	struct event {//事件类型
		IO io; //事件性质:顾客到达/顾客离开
		int serviceNo; //服务柜台编号
		int eventTime; //事件时间
		int customNo; //顾客编号
	};
	struct compare {
		bool operator()(const event& e1, const event& e2)
		{
			return e1.eventTime > e2.eventTime;
		}
	};
	int* service; //服务台
	priority_queue<event, deque<event>, compare> Main; //处理事件队列,按照事件时间优先级排列,事件时间越早,优先级越高
	queue<event> Wait; //排队队列
	int customNum; //顾客数量
	int serviceNum; //服务台数量
	int serviceTimeMax; //最长服务时间
	int serviceTimeMin; //最短服务时间
	int arriveLow;//允许到达最早时间
	int arriveHigh; //允许到达最晚时间
	int searchFree(); //寻找空闲服务台,找不到返回-1
	int currentTime; //模拟中当前时间
public:
	simulator(int sTMax, int sTMin, int cusNum, int serNum,int arL,int arH) {
		customNum = cusNum;
		serviceNum = serNum;
		serviceTimeMax = sTMax;
		serviceTimeMin = sTMin;
		arriveLow = arL;
		arriveHigh = arH;
		service = new int[serNum] {0};//将所有服务台置于空闲状态(0)
		currentTime = 0;
	};
	void Simulation(); //模拟整个排队过程
};

int simulator::searchFree()
{
	for (int i = 0; i < serviceNum; i++)
	{
		if (service[i]==0) return i;
	}
	return -1;
}

void simulator::Simulation()
{
	//产生顾客的服务时间,并存入队列
	event* Eve;
	 Eve=new event[customNum];
	for (int i = 0; i < customNum; i++)
	{
		Eve[i].io = IN;
		Eve[i].eventTime = arriveLow + rand() % (arriveHigh - arriveLow + 1);
		Eve[i].serviceNo = -1; //未服务
		Eve[i].customNo = i;
		Main.push(Eve[i]);
	}

	//开始处理事件
	while (!Main.empty())
	{
		int index;
		event tmp = Main.top();
		Main.pop();
		currentTime = tmp.eventTime; //更新当前时间
		switch (tmp.io) {
		case IN:  //处理到达事件
			if ((index=searchFree())!=-1) //存在空闲服务台,让顾客前往该服务台
			{
				service[index] = 1;
				tmp.serviceNo = index;
				//将时间修改为离开事件,并随机生成其业务服务时间,修改事件时间,重新入队
				tmp.io = OUT;
				int Stime = serviceTimeMin + rand() % (serviceTimeMax - serviceTimeMin + 1);
				tmp.eventTime += Stime;
				Main.push(tmp);
				cout << "当前时间:" << currentTime << endl;
				cout << "服务台" << index << "正在服务顾客" << tmp.customNo << ",服务时长预计" << Stime << "分钟" << endl;
			}
			else {  //不存在空闲服务台,让顾客排队等待
				Wait.push(tmp);
			}
			break;
		case OUT:
			int NO = tmp.serviceNo;
			cout << "当前时间:" << currentTime << endl;
			cout << "服务台" << NO << "完成服务顾客" << tmp.customNo << endl;
			//排队队列非空,让排队队列第一个人接受空出服务台服务
			if (!Wait.empty())
			{
				event tmp2 = Wait.front();
				Wait.pop();
				int Stime = serviceTimeMin + rand() % (serviceTimeMax - serviceTimeMin + 1);
				tmp2.eventTime = currentTime+Stime;
				tmp2.io = OUT;
				tmp2.serviceNo = NO;
				Main.push(tmp2);
				cout << "当前时间:" << currentTime << endl;
				cout << "服务台" << NO << "正在服务顾客" << tmp2.customNo << ",服务时长预计" << Stime << "分钟" << endl;
			}
			else {
				service[NO] = 0;
			}

			break;
		}

	}
	


}

int main()
{
	srand(time(NULL));
	int sTMax, sTMin, cusNum, serNum, arL, arH;
	cout << "输入柜台数:" << endl;
	cin >> serNum;
	cout << "输入顾客数:" << endl;
	cin >> cusNum;
	cout << "输入服务时间下界和上界(单位:分钟)" << endl;
	cin >> sTMin >> sTMax;
	cout << "输入到达时间下界和上界(单位:分钟)" << endl;
	cin >> arL >> arH;

	simulator MySim(sTMax, sTMin, cusNum, serNum,arL,arH);
	MySim.Simulation();
	return 0;

}

 

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

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

相关文章

RabbitMQ--04--发布订阅模式 (fanout)-案例

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 发布订阅模式 (fanout)---案例前言RabbitListener和RabbitHandler的使用 1.通过Spring官网快速创建一个RabbitMQ的生产者项目2.导入项目后在application.yml文件中配…

Python编程-并发编程基础梳理与高级特性案例讲解

Python编程-并发编程基础梳理与高级特性案例讲解 同步、异步通信与IO阻塞 同步&#xff08;Synchronous&#xff09;和异步&#xff08;Asynchronous&#xff09;通信是两种不同的通信方式&#xff0c;涉及到处理任务的时序和相互等待的关系。同步通信简单直观&#xff0c;但可…

Unity 使用TrailRenderer制作拖尾效果

使用TrailRenderer实现拖尾效果&#xff0c;具体操作步骤如下&#xff1a; 1、创建一个空对象 在Unity场景中创建一个空对象 2、添加TrailRenderer组件 选择步骤1创建的空对象&#xff0c;然后在Inspector面板中点击“Add Component”按钮&#xff0c;搜索并添加TrailRende…

蓝桥杯第七届大学B组详解

目录 1.煤球数量&#xff1b; 2.生日蜡烛&#xff1b; 3.凑算式 4.方格填数 5.四平方和 6.交换瓶子 7.最大比例 1.煤球数量 题目解析&#xff1a;可以根据题目的意思&#xff0c;找到规律。 1 *- 1个 2 *** 3个 3 ****** 6个 4 ********** 10个 不难发现 第…

C++入门知识详细讲解

C入门知识详细讲解 1. C简介1.1 什么是C1.2 C的发展史1.3. C的重要性1.3.1 语言的使用广泛度1.3.2 在工作领域 2. C基本语法知识2.1. C关键字(C98)2.2. 命名空间2.2 命名空间使用2.2 命名空间使用 2.3. C输入&输出2.4. 缺省参数2.4.1 缺省参数概念2.4.2 缺省参数分类 2.5. …

Linux - 第三节

改变用户类型 su 仅单纯的进行身份变化 依旧处于普通用户里面 su - 进行重新登录更改身份 退出用exit / ctrld su 用户名 改成成其他身份 对一条命令进行提权 sudo command r:可读 w:可写 x:可执行 -:对应的权限位置&#xff0c;没有权限 去掉所有权限 chmod u…

Qt中继承QCheckBox的类结合QTableWidget实现多选并且每个多选的id都不一样

1.相关描述 继承QCheckBox的类MyCheckBox&#xff0c;利用QTableWidget的setCellWidget方式添加MyCheckBox类的对象 2.相关页面 3.相关代码 mycheckbox.h #ifndef MYCHECKBOX_H #define MYCHECKBOX_H#include <QCheckBox> #include <QObject>class MyCheckBox : pu…

Elvis Presley 英文阅读

Elvis Presley 官方翻译 One of the most popular American singers of thetwentieth century was Elvis Presley. He made theRock & Roll music popular around the world. He sold millions of records and made manysuccessful films, and he helped change the dir…

基于SSM医院病历管理系统

基于SSM医院病历管理系统的设计与实现 摘要 病历管理系统是医院管理系统的重要组成,在计算机技术快速发展之前&#xff0c;病人或者医生如果想记录并查看自己的健康信息是非常麻烦的&#xff0c;因为在以往病人的健康信息通常只保存在自己的病历卡或者就诊报告中&#xff0c;…

TikTok直播专线:解决出海网络问题痛点,提升商业效率

近年来&#xff0c;TikTok作为全球最受欢迎的社交媒体平台之一&#xff0c;成为商家获取商机与市场的重要平台。然而&#xff0c;尽管商家纷纷进入TikTok&#xff0c;试图借助其强大的社交网络进行产品推广和销售&#xff0c;但在TikTok平台进行直播时&#xff0c;往往会遇到网…

何时应用 RAG 与微调

充分发挥 LLM 的潜力需要在检索增强生成&#xff08;RAG&#xff09;和微调之间选择正确的技术。 让我们来看看何时对 LLM、较小的模型和预训练模型使用 RAG 与微调。我们将介绍&#xff1a; LLM 和 RAG 的简要背景RAG 相对于微调 LLM 的优势何时针对不同模型大小对 RAG 进行…

WebGIS 之 vue3+vite+ceisum

1.项目搭建node版本在16以上 1.1创建项目 npm create vite 项目名 1.2选择框架 vuejavaScript 1.3进入项目安装依赖 cd 项目名 npm install 1.4安装cesium依赖 pnpm i cesium vite-plugin-cesium 1.5修改vite.config.js文件 import { defineConfig } from vite import vue fr…

labelme AI 模型运用

一、lebelme 1、界面介绍 点击上图位置&#xff0c;选择对应的模型。这里我每个模型都测试了一下&#xff0c;EfficientSam这个模型最好用&#xff0c;准确率和速度都ok。 2、使用方法 目标框标注方法&#xff1a;点左上角【编辑】-> 【Create Ai-Mask】就可以标志了&…

在 Three.js 中,OBJExporter 是一个用于将 Three.js 中的场景导出为 OBJ 格式的类。

demo案例 在 Three.js 中&#xff0c;OBJExporter 是一个用于将 Three.js 中的场景导出为 OBJ 格式的类。下面是关于它的入参、出参、属性和方法的解释&#xff1a; 类名&#xff1a;OBJExporter 构造函数&#xff1a; THREE.OBJExporter()说明&#xff1a; 创建一个 OBJE…

2d导入人物素材进行分割后设置图层

1、设置分辨率大小 2、相机调整大小&#xff0c;要符合场景 3、选择2D sprite 编辑器 或者 点击这个也行 4、分割图像 5、设置过滤模式 6、图层设置

CQI-17:2021 V2 英文 、中文版。特殊过程:电子组装制造-锡焊系统评审标准

锡焊作为一个特殊的工艺过程&#xff0c;由于其材料特性的差异性、工艺参数的复杂性和过程控制的不确定性&#xff0c;长期以来一直视为汽车零部件制造业的薄弱环节&#xff0c;并将很大程度上直接导致整车产品质量的下降和召回风险的上升。 美国汽车工业行动集团AIAG的特别工…

MySQL 之 数据库操作 及 表操作

&#x1f389;欢迎大家观看AUGENSTERN_dc的文章(o゜▽゜)o☆✨✨ &#x1f389;感谢各位读者在百忙之中抽出时间来垂阅我的文章&#xff0c;我会尽我所能向的大家分享我的知识和经验&#x1f4d6; &#x1f389;希望我们在一篇篇的文章中能够共同进步&#xff01;&#xff01;&…

共享办公室行业面临的最大挑战是什么,未来有哪些可能的发展方向

共享办公室行业虽然发展迅速&#xff0c;但也面临着一些挑战和需要解决的问题。咱们来聊聊这行业的最大挑战和未来可能的发展方向。 面临的最大挑战&#xff1a; 市场竞争加剧&#xff1a;随着共享办公室的火热&#xff0c;越来越多的玩家进入市场&#xff0c;竞争变得异常激烈…

CatalyzeCDN-发现真实IP

简介 CatalyzeCDN用与调用Fofa接口&#xff0c;将查询结果进行整理发现其真实IP 使用说明 在同目录下的config.ini配置好Fofa key ./CatalyzeCDN -h # 查看帮助信息 从根域发现真实IP并列出相应资产 ./CatalyzeCDN -r baidu.com 从IP查询相应资产 ./CatalyzeCDN -i 项目…

VS Code常用前端开发插件和基础配置

VS Code插件安装 VS Code提供了非常丰富的插件功能&#xff0c;根据你的需要&#xff0c;安装对应的插件可以大大提高开发效率。 完成前端开发&#xff0c;常见插件介绍&#xff1a; 1、Chinese (Simplified) Language Pack 适用于 VS Code 的中文&#xff08;简体&#xff…