数据结构:循环队列

news2025/7/7 23:24:42

之前已经做过队列的学习笔记,这一篇写的是循环队列,大部分代码可以继续沿用,某些地方需要作出更改,使其可以实现循环队列的功能。

通俗的总结一下队列的操作,我的思路是将头指针固定不动,然后每一次元素入队就将尾指针后移,每一次出队就把当前头指针指向的元素返回,然后将整个数组整体前移一个位置,尾指针同时减一。而循环队列可以看作将数组围成一个圆形,头指针和尾指针都会根据入队与出队而发生改变,这样就可以省去队列数据结构出队时需要将出队元素除外的其他所有元素整体前移,从而增加了效率。


创建一个CircleQueue类,rear表示尾指针,front表示头指针,size表示用于实现循环队列的数组大小,而arr就是用于实现循环队列的数组。

class CircleQueue{
    private int rear;
    private int front;
    private int size;
    private int[] arr;
}

一、构造循环队列

与普通队列不同的是,头指针front与尾指针rear的初始值都设置为0,并且由于循环队列中需要空出一个位置用于判断和保持操作一致,因此数组实际大小应该比用户期待队列的大小大1,因此size的值应该为用户期待队列大小的值maxsize加一。

    public CircleQueue(int maxsize){
        size = maxsize + 1;
        arr = new int[size];
        rear = 0;
        front = 0;
    }

二、判断循环队列是否已满

由于队列是循环的,因此判满的方法也需要更改。当尾指针+1的值对队列大小size取模的结果值刚好为头指针的值则代表队列已满。(用文字描述难免有些晦涩,在草稿画个环作数组运算一下便于理解。)

    public boolean isFull(){
        return (rear + 1) % size == front;
    }

三、判断循环队列是否已空

当尾指针rear与头指针front重叠时,即证明当前队列为空。(队列已满应该为两指针相邻,所以才有判满的那条公式(rear+1)%size=front。这也是循环数组需要预留一个位置用于判断,否则判满与判空的条件很可能就相同从而无法判断了。)

    public boolean isEmpty(){
        return rear == front;
    }

四、元素入队

入队前,判满操作还是不能少。通过判断后,将出入元素element保存在rear所指向的位置,然后rear指针按照公式(rear +1) % size移动(为了实现循环,需要取模操作)。最后打印“入队成功提示”作提示。

    public void addQueue(int element){
        if(isFull()){
            //throw new RuntimeException("队列已满!");
            System.out.println("队列已满!");
            return;
        }
        arr[rear] = element;
        rear = (rear +1) % size;
        System.out.println("入队成功!");
    }

五、元素出队

出队前同样需要判空操作。接着先将出队元素用临时变量ans保存着,然会按公式移动front指针,让front后退一个位置,最后返回出队元素。

    public int getQueue(){
        if(isEmpty()){
            throw new RuntimeException("队列已空!");
        }
        int ans = arr[front];
        front = (front + 1) % size;
        return ans;
    }

六、展示队列

循环队列的展示队列操作有点点绕。第一步判空操作(可有可无)。getSzie方法就是计算从rear到size之间的距离,因为是循环队列,所以也需要进行取模运算。计算后,在showQueue方法中,以front为起点,直至front加上相距距离。但是在打印时不能用i作为数组下标,这样很可能会出现数组越界。为了使下标正确,还需要对i进行取模运算,运算结果才是正确的数组下标。最后实现遍历循环队列,输出打印。

    public void showQueue(){
        if(isEmpty()){
            System.out.println("队列为空,没有数据~");
            return;
        }
        System.out.println("队列展示:");
        for(int i = front ; i < front + getSize(); i++){
            System.out.println(arr[i % size]);
            //System.out.println(arr[i]);
        }
//        System.out.println("front:" + front);
//        System.out.println("getsize:" + getSize());
    }
    public int getSize(){
        return (rear + size -front) % size;
    }

七、获取队首元素

这个方法与普通队列中完全相似,同样先判空,若符合要求则返回队首元素。在此不再赘述。

    public void rearQueue(){
        if(isEmpty()){
            System.out.println("队列为空,peek不了啦~");
            return;
        }
        System.out.println("队头元素是:" + arr[front]);
    }

八、界面

界面保持一致,不需要作更改。将代码再放一次在这里。

public class CircleArray {
    public static void main(String[] args){
        Scanner input = new Scanner(System.in);
        boolean flag = true;
        char key = ' ';

        System.out.print("请输入队列大小:");
        Scanner inputs = new Scanner(System.in);
        int size = inputs.nextInt();
        CircleQueue AQ = new CircleQueue(size);

        while(flag){
            System.out.println("---------------------------");
            System.out.println("|(1)退出程序:e(exit)     |");
            System.out.println("|(2)元素入队:a(add)      |");
            System.out.println("|(3)元素出队:g(get)      |");
            System.out.println("|(4)查询队头:p(peek)     |");
            System.out.println("|(5)展示队列:s(show)     |");
            System.out.println("---------------------------");
            System.out.print("【输入字母选择操作】");
            key = input.next().charAt(0);
            switch (key){
                case 'a':
                    System.out.print("入队元素:");
                    int element = input.nextInt();
                    AQ.addQueue(element);
                    break;
                case 'g':
                    System.out.println("出队元素是:" + AQ.getQueue());
                    break;
                case 'p':
                    AQ.rearQueue();
                    break;
                case 's':
                    AQ.showQueue();
                    break;
                case 'e':
                    System.out.println("成功退出...");
                    flag = false;
                    break;
                default:
                    System.out.println("输入不合法,请重新输入~");
            }

        }
    }

}

完整代码:

import java.util.Scanner;

public class CircleArray {
    public static void main(String[] args){
        Scanner input = new Scanner(System.in);
        boolean flag = true;
        char key = ' ';

        System.out.print("请输入队列大小:");
        Scanner inputs = new Scanner(System.in);
        int size = inputs.nextInt();
        CircleQueue AQ = new CircleQueue(size);

        while(flag){
            System.out.println("---------------------------");
            System.out.println("|(1)退出程序:e(exit)     |");
            System.out.println("|(2)元素入队:a(add)      |");
            System.out.println("|(3)元素出队:g(get)      |");
            System.out.println("|(4)查询队头:p(peek)     |");
            System.out.println("|(5)展示队列:s(show)     |");
            System.out.println("---------------------------");
            System.out.print("【输入字母选择操作】");
            key = input.next().charAt(0);
            switch (key){
                case 'a':
                    System.out.print("入队元素:");
                    int element = input.nextInt();
                    AQ.addQueue(element);
                    break;
                case 'g':
                    System.out.println("出队元素是:" + AQ.getQueue());
                    break;
                case 'p':
                    AQ.rearQueue();
                    break;
                case 's':
                    AQ.showQueue();
                    break;
                case 'e':
                    System.out.println("成功退出...");
                    flag = false;
                    break;
                default:
                    System.out.println("输入不合法,请重新输入~");
            }

        }
    }

}

class CircleQueue{
    private int rear;
    private int front;
    private int size;
    private int[] arr;

    //构造
    public CircleQueue(int maxsize){
        size = maxsize + 1;
        arr = new int[size];
        rear = 0;
        front = 0;
    }

    //判满
    public boolean isFull(){
        return (rear + 1) % size == front;
    }

    //判空
    public boolean isEmpty(){
        return rear == front;
    }

    //入队
    public void addQueue(int element){
        if(isFull()){
            //throw new RuntimeException("队列已满!");
            System.out.println("队列已满!");
            return;
        }
        arr[rear] = element;
        rear = (rear +1) % size;
        System.out.println("入队成功!");
    }

    //出队
    public int getQueue(){
        if(isEmpty()){
            throw new RuntimeException("队列已空!");
        }
        int ans = arr[front];
        front = (front + 1) % size;
        return ans;
    }

    //展示队列
    public void showQueue(){
        if(isEmpty()){
            System.out.println("队列为空,没有数据~");
            return;
        }
        System.out.println("队列展示:");
        for(int i = front ; i < front + getSize(); i++){
            System.out.println(arr[i % size]);
            //System.out.println(arr[i]);
        }
//        System.out.println("front:" + front);
//        System.out.println("getsize:" + getSize());
    }
    public int getSize(){
        return (rear + size -front) % size;
    }

    //peek
    public void rearQueue(){
        if(isEmpty()){
            System.out.println("队列为空,peek不了啦~");
            return;
        }
        System.out.println("队头元素是:" + arr[front]);
    }

    //

}

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

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

相关文章

模拟实现string

第一部分&#xff1a;构造&#xff0c;析构&#xff0c;拷贝构造&#xff0c;赋值重载&#xff0c;打印函数这几个大头写出来先 string类框架 namespace xxx { class string { public: // //private: char* _str; size_t _size; size_t _capacity;const static size_t npos -…

计算机视觉——python在一张图中画多条ROC线

在验证分类算法的好坏时&#xff0c;经常需要用到AUC曲线&#xff0c;而在做不同分类模型的对比实验时&#xff0c;需要将不同模型的AUC曲线绘制到一张图里。 计算机视觉——python在一张图中绘制多个模型的对比ROC线1. 小型分类模型对比&#xff0c;可以直接调用的2. 大型的CN…

七夕,程序员教你5个表白代码,2分钟学会,牢牢主抓她的心

七夕。一个有人欢喜有人愁的节日&#xff0c;虽然对一些单身人士不太友好&#xff0c;但还有不少人都在等这个节日进行表白。毕竟这个日子的成功率会高一些。 情人节少不了送花送礼物&#xff0c;作为一个程序员&#xff0c;当然不会在送什么礼物上给你指点一二&#xff0c;但…

适合骑车时候戴的耳机怎么选,列举五款在骑行佩戴的耳机推荐

相信大多数人在运动的过程中都会感觉到枯燥无力的感觉&#xff0c;为此也一直在寻找一些能够让我们在运动中保持最初的热诚&#xff0c;在最终的选择上&#xff0c;绝大多数都是选择了耳机&#xff0c;因为耳机能够产生美妙的音乐&#xff0c;将我们运动的枯燥做进一步的抵消&a…

【附源码】Python计算机毕业设计网络教育平台设计

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

【HMS Core】定位服务无法获取街道信息问题

问题描述&#xff1a; 华为HMS-定位服务无法获取省市街道信息 背景&#xff1a;在手机上集成华为定位服务的功能运行后&#xff0c;只能获取到经纬度&#xff0c;无法得到具体地址(城市/街道/建筑)。 配置环境&#xff1a;&#xff08;1&#xff09;手机型号&#xff1a;Red…

c语言实现通讯录(用三种方法来实现一个属于你的通讯录)

前沿&#xff1a; &#xff5e;&#xff5e;在一个周前&#xff0c;我用c语言实现了一个银行账号的功能&#xff0c;在总结当中我说了要实现一个通讯录&#xff0c;来实现人员的增删插改功能&#xff0c;而现在就是我实现它的时候&#xff01;&#xff01;&#xff0c;本文呢小…

基于python命令流及代码的Plaxis自动化建模

有限单元法在岩土工程问题中应用非常广泛&#xff0c;很多商业软件如Plaxis/Abaqus/Comsol等都采用有限单元解法。在使用各大软件进行数值模拟建模的过程中&#xff0c;您是否发现GUI界面中重复性的点击输入工作太繁琐&#xff1f;从而拖慢了设计或方案必选进程&#xff1f; 本…

SpringBoot八种bean的加载方式一学就会

目录 文章目录[toc]第一种bean的加载方式-配置文件第二种加载bean方式-注解和扫描创建第三方的bean对象第三种加载bean方式-不使用配置文件扩展-bean的加载方式扩展FactoryBean<>扩展-ImportResource导入配置文件扩展-proxyBeanMethods属性-产生代理对象第四种加载bean方…

一文读懂,python实现常用的数据编码和对称加密

相信很多使用 python 的小伙伴在工作中都遇到过&#xff0c;对数据进行相关编码或加密的需求&#xff0c;今天这篇文章主要给大家介绍对于一些常用的数据编码和数据加密的方式&#xff0c;如何使用 python 去实现。话不多说&#xff0c;接下来直接进入主题&#xff1a; 1、bas…

Windows OpenGL ES 图像灰度图

目录 一.OpenGL ES 图像灰度图 1.原始图片2.效果演示 二.OpenGL ES 图像灰度图源码下载三.猜你喜欢 零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 基础 零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 特效 零基础 Open…

Redis添加至windows服务

1.进入redis文件夹 打开redis所在文件夹&#xff0c;在资源管理器地址栏输入cmd&#xff0c;打开管理员命令提示符窗口。 2.配置服务参数 打开redis.windows.conf文件&#xff0c;找到“requirepass”配置项&#xff0c;此处设置redis密码&#xff0c;如果不需要密码&#xff0…

Python如何使用HanNLP工具

Python如何使用HanNLP工具目录系统安装配置 JDK 1.8系统安装 Visual C 2015下载 HanNLP 包测试HanNLP工具目录 目标&#xff1a;使用pycharm调用HanNLP工具完成对文本的分词、自动摘要、关键词提取等任务。 系统安装配置 JDK 1.8 1、windows环境下载 JDK 1.8 2、安装 JDK 1.…

Oracle PrimaveraUnifier空间管理器(Space Manager)

目录 前言 介绍 前言 在Oracle Primavera Unifier设施和资产管理基础产品中除了业务流程及配置管理器之外&#xff0c;其预配置设计还包含本文要介绍的空间管理器&#xff0c;即Space Manager 在Unifier中&#xff0c;空间管理器是用户可以执行设施管理任务的地方。空间管…

xv6---Lab1: Xv6 and Unix utilities

目录 参考资料&#xff1a; 1.1进程和内存 1.2 I/O 和文件描述符 1.3管道 源码&#xff1a; 调试环境搭建 sleep PingPong primes find xargs 参考资料&#xff1a; Lab: Xv6 and Unix utilities xv6-book翻译(自用&#xff09;第一章 - 知乎 1.1进程和内存 一个xv…

html中css的基础学习

小李胖了吗 I 都说秋天适合思念&#xff0c;其实更适合见面【小李胖了吗 I 都说秋天适合思念&#xff0c;其实更适合见面】 https://www.bilibili.com/video/BV19g411B7uL/?share_sourcecopy_web&vd_source385ba0043075be7c24c4aeb4aaa73352 通过本博文的学习&#xff0c…

常见的软件测试面试题,千万别答错了

软件测试的童鞋们&#xff0c;在面试测试工作时&#xff0c;一定遇到面试官问过这个问题&#xff1a; 软件测试的目的意义是什么&#xff1f;大家是怎么回答的呢&#xff1f;如果这个问题回答好了&#xff0c;说明你对软件测试工作的价值与意义了如指掌。 有经验的测试人员可…

[数据结构]栈和队列面试题解析

作者&#xff1a; 华丞臧. 专栏&#xff1a;【数据结构】 各位读者老爷如果觉得博主写的不错&#xff0c;请诸位多多支持(点赞收藏关注)。如果有错误的地方&#xff0c;欢迎在评论区指出。 推荐一款刷题网站 &#x1f449; LeetCode刷题网站 文章目录一、有效括号题目描述解题思…

freemarker+yml介绍 以及freemarker与JSP的区别

目录 1. freemarker介绍 2. freemarker使用步骤 2.1 在pom.xml引入freeMarker的依赖包 2.2 在springboot中添加freemarker配置 2.3 编写模板文件*.ftl(当做jsp使用即可) ​编辑 2.4 访问控制器后进行页面跳转 3. freemarker常用语法 3.1 取值 3.2 条件 3.3 循环 3…

超赞:不愧是“阿里内部Redis学习笔记”从头到尾,全是精华

近几年&#xff0c;随着移动互联网的飞速发展&#xff0c;我们享受着整个社会的技术进步带来的便利&#xff0c;但同时也给从业者带来了如何保证项目的高并发、低延时的技术挑战&#xff0c;相应的互联网技术也随之发生了重大变革&#xff0c;NoSQL技术得到了蓬勃的发展。 Red…