当前位置: 首页 > news >正文

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

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

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

基于普通线性表实现优先级队列,入队和出队中必有一个时间复杂度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;}//否则下滤结束elsebreak;	//否则选择左孩子结点上滤
}

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;}elsebreak;	

建堆过程可以表示为

//建堆 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(); //寻找空闲服务台,找不到返回-1int 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;}

 

相关文章:

  • 【面试专题】MySQL
  • 网络基础二补充——json与http协议
  • 网络编程的学习1
  • 火车头通过关键词采集文章的原理
  • Linux中断管理:(一)中断号的映射
  • JavaScript高级 —— 学习(四)
  • 「MySQL」索引事务
  • electron打包桌面版.exe之vue项目踩坑(vue3+electron 解决打包后首页打开空白,打包后路由不跳转及请求不到后端数据等问题)
  • 基于OrangePi Zero2的智能家居项目(开发阶段)
  • EMD关于信号的重建,心率提取
  • Linux之进程间通信
  • 用JSch实现远程传输文件并打包成jar
  • (一)kafka实战——kafka源码编译启动
  • 睿尔曼超轻量仿人机械臂之复合机器人底盘介绍及接口调用
  • Linux|如何管理多个Git身份
  • [译]前端离线指南(上)
  • 【跃迁之路】【669天】程序员高效学习方法论探索系列(实验阶段426-2018.12.13)...
  • github指令
  • JavaScript HTML DOM
  • javascript从右向左截取指定位数字符的3种方法
  • java第三方包学习之lombok
  • orm2 中文文档 3.1 模型属性
  • webgl (原生)基础入门指南【一】
  • 阿里云应用高可用服务公测发布
  • 基于web的全景—— Pannellum小试
  • 前端面试之闭包
  • AI算硅基生命吗,为什么?
  • # Java NIO(一)FileChannel
  • (4)STL算法之比较
  • (C语言)fgets与fputs函数详解
  • (附源码)springboot炼糖厂地磅全自动控制系统 毕业设计 341357
  • (附源码)SSM环卫人员管理平台 计算机毕设36412
  • (附源码)计算机毕业设计SSM教师教学质量评价系统
  • (免费领源码)Java#ssm#MySQL 创意商城03663-计算机毕业设计项目选题推荐
  • .bat批处理(七):PC端从手机内复制文件到本地
  • .NET Core实战项目之CMS 第一章 入门篇-开篇及总体规划
  • .net 后台导出excel ,word
  • .NET 中让 Task 支持带超时的异步等待
  • .NET/C# 的字符串暂存池
  • .NET/C# 利用 Walterlv.WeakEvents 高性能地中转一个自定义的弱事件(可让任意 CLR 事件成为弱事件)
  • .net中生成excel后调整宽度
  • @RequestMapping处理请求异常
  • [ IOS ] iOS-控制器View的创建和生命周期
  • [ linux ] linux 命令英文全称及解释
  • [ 蓝桥杯Web真题 ]-Markdown 文档解析
  • [16/N]论得趣
  • [asp.net core]project.json(2)
  • [C puzzle book] types
  • [C++ 从入门到精通] 12.重载运算符、赋值运算符重载、析构函数
  • [HNOI2015]实验比较
  • [ndss 2023]确保联邦敏感主题分类免受中毒攻击
  • [NEWS] J2SE5.0来了
  • [oeasy]python0002_终端_CLI_GUI_编程环境_游戏_真实_元宇宙
  • [PostgreSQL的 SPI_接口函数]
  • [python] RRT快速拓展随机树