队列的实现

news2025/1/10 3:24:09

队列

简介

  队列是一种线性表的特殊形式,特殊之处在于它只允许在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作,和栈一样,队列是一种操作受限制的线性表。进行插入操作的端称为队尾,进行删除操作的端称为队头。队列中没有元素时,称为空队列。

  队列的数据元素又称为队列元素。在队列中插入一个队列元素称为入队,从队列中删除一个队列元素称为出队。因为队列只允许在一端插入,在另一端删除,所以只有最早进入队列的元素才能最先从队列中删除,故队列具有先进先出(FIFO)的特点,可以用来实现各种数据缓存、任务队列、消息队列等应用场景。

类型

  队列一般分为两种,一种是单调队列,也是日常生活中最常见的队列,无论是在食堂还是在超市都随处可见。
在这里插入图片描述

  另一种就是循环队列,也就是队头与队尾在同一个位置(此时队列为空),存储元素时队尾向后绕圈,直到队头与队尾只相差一个距离时停止(是队尾-队头=1,而不是队头-队尾=1)。
在这里插入图片描述

代码实现

  说了这么多,那么,如何用代码来实现队列呢?

  下面将用链表的方式实现单调队列,用模拟数组的方式实现循环队列

单调队列

首先明确需要实现的几个功能:

  1. 初始化队列
  2. 判断队列是否为空(方便后续出队判断)
  3. 入队
  4. 出队
  5. 遍历队列元素

然后定义两个结构体,一个是节点(这里使用的是双链表的形式,方便入队和出队操作),一个是队列本体

// 结构体定义节点
typedef struct Node{
    int data; //数据域
    struct Node *pNext; //指针域,指向下一个节点
    struct Node *pBefore; //指针域,指向上一个节点
}node,*pnode;
// 结构体定义队列
typedef struct Queue{
    node *front; //队头
    node *rear; //队尾
}Queue;

接着就是函数声明

//函数声明
void init(Queue *queue); //初始化队列
bool isEmpty(Queue *queue); //判断队列是否为空
bool offer(Queue *queue,int val); //数据入队
int poll(Queue *queue); //数据出队,并返回其中的数据
void traverse(Queue *queue); //遍历队列元素

然后逐个函数进行实现

初始化队列

void init(Queue *queue){
    queue->front = (pnode) malloc(sizeof (node));
    queue->rear = (pnode) malloc(sizeof (node));
    queue->rear->pNext = queue->front;
    queue->rear->pBefore = NULL;
    queue->front->pBefore = queue->rear;
    queue->front->pNext=NULL;
}

但看代码有点绕,但如果画个图就很简单明了了。
在这里插入图片描述

判断队列是否为空

bool isEmpty(Queue *queue){
    if(queue->rear->pBefore==queue->front){ //如果队尾的下一个节点为队头,则证明队列为空
        return true;
    }
    return false;
}

入队

bool offer(Queue *queue,int val){
    //添加新的节点
    pnode pnew = (pnode) malloc(sizeof (node));
    pnew->data = val;
    pnew->pNext = queue->rear->pNext;
    queue->rear->pNext->pBefore = pnew;
    queue->rear->pNext = pnew;
    pnew->pBefore = queue->rear;
    return true;
}

出队

int poll(Queue *queue){
    if(isEmpty(queue)){ //如果队列为空,则无法出队
        return -1;
    }
    int val = queue->front->pBefore->data;
    pnode q = queue->front->pBefore;
    queue->front->pBefore = queue->front->pBefore->pBefore;
    queue->front->pBefore->pNext = queue->front;
    q = NULL;
    free(q);
    return val;
}

遍历队列

void traverse(Queue *queue){
    if(isEmpty(queue)){ //栈为空,无法遍历
        return ;
    }
    pnode q = queue->front->pBefore;
    while(q!=queue->rear){//从队头遍历到队尾,相当于从第一个人开始往后报数
        printf("%d ",q->data);
        q = q->pBefore;
    }
    printf("\n");
}

最后,在main函数中进行测试

int main(){
    //定义一个队列并初始化
    Queue queue;
    init(&queue);
    //入队操作
    offer(&queue,1);
    offer(&queue,2);
    offer(&queue,3);
    offer(&queue,4);
    offer(&queue,5);
    //遍历队列
    traverse(&queue);
    //
    int num = poll(&queue);
    printf("队头元素为:%d",num);
    return 0;
}

运行结果

在这里插入图片描述

最后,附上完整的代码

#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
// 结构体定义节点
typedef struct Node{
    int data; //数据域
    struct Node *pNext; //指针域,指向下一个节点
    struct Node *pBefore; //指针域,指向上一个节点
}node,*pnode;
// 结构体定义队列
typedef struct Queue{
    node *front; //队头
    node *rear; //队尾
}Queue;
int main(){
    //定义一个队列并初始化
    Queue queue;
    init(&queue);
    //入队操作
    offer(&queue,1);
    offer(&queue,2);
    offer(&queue,3);
    offer(&queue,4);
    offer(&queue,5);
    //遍历队列
    traverse(&queue);
    //
    int num = poll(&queue);
    printf("队头元素为:%d",num);
    return 0;
}
void init(Queue *queue){
    queue->front = (pnode) malloc(sizeof (node));
    queue->rear = (pnode) malloc(sizeof (node));
    queue->rear->pNext = queue->front;
    queue->rear->pBefore = NULL;
    queue->front->pBefore = queue->rear;
    queue->front->pNext=NULL;
}
bool isEmpty(Queue *queue){
    if(queue->rear->pBefore==queue->front){ //如果队尾的下一个节点为队头,则证明队列为空
        return true;
    }
    return false;
}
bool offer(Queue *queue,int val){
    //添加新的节点
    pnode pnew = (pnode) malloc(sizeof (node));
    pnew->data = val;
    pnew->pNext = queue->rear->pNext;
    queue->rear->pNext->pBefore = pnew;
    queue->rear->pNext = pnew;
    pnew->pBefore = queue->rear;
    return true;
}
int poll(Queue *queue){
    if(isEmpty(queue)){ //如果队列为空,则无法出队
        return -1;
    }
    int val = queue->front->pBefore->data;
    pnode q = queue->front->pBefore;
    queue->front->pBefore = queue->front->pBefore->pBefore;
    queue->front->pBefore->pNext = queue->front;
    q = NULL;
    free(q);
    return val;
}
void traverse(Queue *queue){
    if(isEmpty(queue)){ //栈为空,无法遍历
        return ;
    }
    pnode q = queue->front->pBefore;
    while(q!=queue->rear){//从队头遍历到队尾,相当于从第一个人开始往后报数
        printf("%d ",q->data);
        q = q->pBefore;
    }
    printf("\n");
}

循环队列

根据上面的分析,这里就不过多赘述。只是循环队列比单调队列多了一个判断是否已满的功能(判断是否还能入队)

结构体定义一个队列

typedef  struct Queue{ //结构体定义队列
    int *pBase; // 数组模拟数据域
    int front; // 队头
    int rear; // 队尾
}Queue;

函数声明

void init(Queue *queue);//初始化队列
bool offer(Queue *queue,int val);//入队
void traverse(Queue *queue);//遍历并输出队列
bool isFull(Queue *queue);//判断队列是否满
bool poll(Queue *queue,int *pVal);//出队
bool isEmpty(Queue *queue);//判断队列是否空

初始化队列

void init(Queue *queue){
    queue->pBase = (int *)malloc(sizeof(int)*6); // 这里只给了6个int的空间
    queue->front = 0;
    queue->rear = 0;
}

判断队列是否为空

bool isEmpty(Queue *queue){
    if(queue->front == queue->rear){ //如果队头与队尾的下标相同则说明队列为空
        return true;
    }
    return false;
}

判断队列是否已满

bool isFull(Queue *queue){
    if((queue->rear+1)%6 == queue->front){
        return true;
    }
    return false;
}

这里有个公式判断是否队列已满:(队尾下标+1)%队列长度==队头下标

如果该公式为true,则证明队列已满,否则为未满。

入队

bool offer(Queue *queue,int val){
    if(isFull(queue)){
        printf("队列已满!\n");
        return false;
    }
    queue->pBase[queue->rear] = val;
    queue->rear = (queue->rear+1)%6;
}

出队

bool poll(Queue *queue, int *pVal){
    if(isEmpty(queue)){
        return false;
    }
    *pVal = queue->pBase[queue->front];
    queue->front = (queue->front+1)%6;
    return true;
}

遍历队列

void traverse(Queue *queue){
    int i = queue->front;
    while(i!=queue->rear){
        printf("%d  ",queue->pBase[i]);
        i = (i+1)%6;
    }
    printf("\n");
}

注意i的值,因为此时队头下标不一定比队尾小,所以在遍历的时候要对队列长度取模。

测试

最后,加上main主函数进行测试

int main() {
    int val;
    Queue queue;
    //初始化队列
    init(&queue);
    offer(&queue,1);
    offer(&queue,2);
    offer(&queue,3);
    offer(&queue,4);
    offer(&queue,5);
    offer(&queue,6);
    //此时队列已满
    offer(&queue,7);
    offer(&queue,8);
    //遍历队列
    traverse(&queue);
    if(poll(&queue,&val)){ //出队
        printf("出队成功,队列出队的元素为%d\n",val);
    }else{
        printf("出队失败!\n");
    }
    traverse(&queue);
    return 0;
}

执行结果
在这里插入图片描述

到此,单调队列和循环队列已经学习完毕~

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

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

相关文章

信息安全复习五:数据加密标准(DES)

一、本章梗概 1.主要内容&#xff1a;分组密码、分组密码用到的关键技术和结构、对称密钥密码典型算法DES 2.思考问题&#xff1a; ①按照明文被处理的形式&#xff0c;DES属于标准的什么密码 ②根据密钥的使用数量&#xff0c;DES属于标准的什么密码 3.内容回顾&#xff1a; …

力扣sql中等篇练习(十二)

力扣sql中等篇练习(十二) 1 产品销售分析 ||| 1.1 题目内容 1.1.1 基本题目信息 1.1.2 示例输入输出 1.2 示例sql语句 # Write your MySQL query statement below SELECT s1.product_id,s1.first_year,s2.quantity,s2.price FROM (SELECT product_id,Min(year) first_yearF…

基于趋动云的chatGLM-6B模型的部署

首先根据官方示例教程&#xff0c;学会怎么创建项目&#xff0c;怎么使用数据&#xff0c;怎么进入开发环境&#xff0c;以及了解最重要的2个环境变量&#xff1a; 这个是进入开发环境以后的代码目录 $GEMINI_CODE 这个是引用数据集后&#xff0c;数据集存放的路径 $GEMINI_DA…

学生信息管理系统简易版(文件读写操作)

功能模块 具体功能如下&#xff1a; 添加学生信息修改学生信息&#xff08;按学号&#xff09;排序&#xff08;分别按总分升序、降序、以及按姓名升序&#xff09;查找学生&#xff08;按学号&#xff09;删除学生查看所有学生信息 数据结构体设计 本表设计一个学生信息的结…

computed和watch

computed: 写法&#xff1a; import {computed} from vue setup(){ --- //计算属性—简写 let fullName computed(()>{ return person.firstName - person. lastName}) //计算属性-完整 let fullName computed({ get(){ return person.firstName - person. lastName},…

中小企业真的需要CRM吗?

如果你的企业没有CRM客户关系管理系统&#xff0c;企业主需要问问自己&#xff0c;他们将利用什么来扩展业务。福布斯进行的研究恰当地表明&#xff0c;充分利用CRM系统的企业可以将销售额提高29%。 中小企业定期产生大量客户&#xff0c;这可能会难以管理。这正是CRM系统在有…

Esxi8.0安装Ubuntu系统教程

本篇教程主要教大家怎么在ESXi8.0虚拟机上安装Ubuntu系统&#xff0c;首先安装Ubuntu需要准备一个ISO系统镜像文件&#xff0c;我们可以去Ubuntu官网下载。 Ubuntu官网&#xff1a;https://ubuntu.com/download/desktop 点击【Download】即可下载Ubuntu的ISO系统镜像文件 ESXi…

必学宝典 黑马《最新JavaWeb开发教程》上线

对于程序员&#xff0c;所在的行业更迭实属过快&#xff0c;如果是为了找一份好工作&#xff0c;学技术前一定要先了解技术在市场中的需求情况。不然等你学完之后&#xff0c;才发现自己学了已被淘汰、过时的技术&#xff0c;白白浪费了宝贵的学习时间&#xff0c;后悔都来不及…

网络编程代码实例:传输控制协议(TCP)简单版

文章目录 前言代码仓库内容代码&#xff08;有详细注释&#xff09;server.cclient.cMakefile 结果总结参考资料作者的话 前言 网络编程代码实例&#xff1a;传输控制协议&#xff08;TCP&#xff09;简单版。 代码仓库 yezhening/Environment-and-network-programming-examp…

【备份】使用ubuntu一个月,记录的问题和解决方案

备份一下 40.ANSI 转义码39.终端鼠标38.键盘映射37.端口36.nmap扫描35.磁盘管理34.关机默认等待时间33.HackBGRT电脑logo32.lsblk31.update-initramfs30.fastGithub29.rename28.设置休眠27.小鱼ROS26.查看磁盘25.wmctrl24.Typora图片存储23.ssh远程登录xrdp桌面连接 22.油猴插件…

[渗透教程]-013-网络实体标识及网络监听

文章目录 1.网络实体标识2. 常见的代理服务2.1 虚拟专用网络VPN2.2socks代理3.网络监听3.1被动监听3.2主动监听3.3 监听工具3.3.2dsniff1.网络实体标识 2. 常见的代理服务 2.1 虚拟专用网络VPN VPN 是企业网在因特网等公共网络的延伸,我们可以把它理解成是虚拟出来的企业内部…

实验五~JDBC数据库访问与DAO设计模式

1. 使用传统JDBC方法与Model 1模型通过JSP页面访问数据库。 【步骤1】在MySQL数据库test中创建books表&#xff0c;其结构如下&#xff1a; 创建数据库MySQL代码与插入数据代码 drop database if exists web_test; create database web_test character set utf8mb4;use web_…

oralce数据库定时备份

windows服务器中&#xff0c;定时备份数据库&#xff08;全量备份&#xff09;。 创建一个bat文件&#xff0c;内容为&#xff1a; set dd%date:~8,2% set mm%date:~5,2% set yy%date:~0,4% set folder%yy%-%mm%-%dd% md D:\db_backup\%folder% exp db_name/db_pwd127.0.0.1/or…

为了写markdown文件,Windows下的noevim配置

vim安装插件 在网页上写CSDN文章有诸多不便&#xff0c;感觉vim很好用&#xff0c;可是vim中没有预览功能。按照网上找到的教程安装插件&#xff0c;将以下代码复制的到_vimrc文件中。 set rtp$VIM/vimfiles/bundle/Vundle.vim/ call vundle#begin(./vimfiles/bundle/)"…

4.3调整基类成员在派生类中的访问属性的方法

同名成员 在定义派生类的时候&#xff0c;C语言允许派生类与基类中的函数名相同。如果在派生类中定义了与基类中相同的成员&#xff0c;则称派生类成员覆盖了基类的同名成员&#xff0c;在派生类中使用这个名字意味着访问在派生类中重新说明的成员。为了在派生类中使用基类的同…

瑞吉外卖-项目笔记

文章目录 1.业务开发day011.软件开发整体介绍2.项目整体介绍:star:3.开发环境搭建4.登录功能&#xff1a;star4.1代码实现 5.退出功能6.页面效果出现 day021.完善登录功能2.新增员工功能 1.业务开发 day01 1.软件开发整体介绍 2.项目整体介绍⭐️ 后端&#xff1a;管理菜品和…

车载以太网 - SomeIP - 协议用例 - Format_02

目录 11、验证SomeIP-SD Resvered字段设置为0x000000 13、验证SomeIP-SD Entry Type类型设置为0x01(提供服务)

基于脉搏波信号和人工智能方法的应用

目录 一、引言 二、脉搏波信号基本原理和特点 2.1 脉搏波信号的产生和传播 2.2 脉搏波信号的特征参数 三、人工智能技术在脉搏波信号处理中的应用 3.1 传统机器学习方法 3.2 深度学习方法 四、基于深度学习的脉搏波信号处理方法 4.1 数据预处理 4.2 深度学习模型构建…

写题总结1

先把自己写完的总结一下&#xff1a; 题目一&#xff1a; 猫儿园的告示牌上贴着 ab 大小的矩形广告纸。猫猫对广告不感兴趣&#xff0c;她想知道能否用 cd 的矩形白纸完全覆盖这个广告。猫猫可以对白纸进行平移、旋转&#xff0c;但不能折叠或撕开等。如果可以完全覆盖输出 YE…

【音视频第17天】RTSP、RTMP协议初识

被叫去搞直播了&#xff0c;悲喜交加。先学习一下基本的技术栈&#xff0c;RTSP RTMP HTTP 先简单随便看看吧。 目录 什么是流媒体协议RTMPRTMP 工作原理 RTSPRTSP 工作原理 RTMP 与 RTSP 区别详细看看RTSP简介RTSP交互流程OPTIONSDESCRIBESETUPPLAYPAUSESET_PARAMETERGET_PAR…