Java基础之数组模拟循环队列

一、队列简介

队列是一个有序列表,遵循“先入先出”的原则,即先存入队列的数据要先取出,后存入的数据后取出。

队列有两种存储表示,顺序表示和链式表示。顺序表示可以用数组来实现。

二、数组模拟队列

用数组模拟队列时,设两个值front=0,rear=0。front表示队列首部第一个数据所在位置,rear表示尾部最后一个数据的下一个位置。

将数据插入数组队列时(入队),从尾部进行插入,即array[rear] = value,同时rear后移,rear++。取出数据时(出队),从头部取出数据,value = array[front],同时front后移front++。

当front=0,rear=maxSize(数组的最大长度),队列满,不能再存入数据。

这时如果执行出队操作,又有空间可以存储,但继续插入数据,会出现溢出现象,即因数组越界而导致程序非法操作错误。而实际上空间并未占满,所以称这种现象为“假溢出”。这是由“队尾入队,队头出队”的限制操作所造成的。

如何解决“假溢出”的问题呢?

答案就是循环队列。

三、数组模拟循环队列

将顺序队列变为一个环状空间,front、rear和队列元素之间的关系不变,只是在循环队列中,front、rear依次后移加1的操作可用“模”运算来实现。

通过取模,front、rear就可以在顺序表空间内以头尾衔接的方式“循环”移动。

元素出队后,front = (front+1)%maxSize ;
元素入队后,rear = (rear+1)%maxSize 。

(maxSize为数组队列的最大长度)

例如,队列最大长度为4,当rear=3时,存入数据,array[3]=value,rear后移一位,如果没有取模运算,则rear=4,这时继续插入数据,array[4]越界。

如果进行取模运算,rear = (rear+1)%maxSize ,这时rear=0,rear又重新回到了0的位置。这样的运算,使得rear的值在0、1、2、3之间循环。

front的值同理。

写了这么多字,感觉还不如看代码来得更容易理解一点。

四、代码实现

数组模拟循环队列

package com.ArrayQueue;

import java.util.Scanner;

public class CircleArrayQueueDemo {
    public static void main(String args[]){
        int maxSize = 4;
        CircleArrayQueue queue = new CircleArrayQueue(maxSize);
        Scanner scanner = new Scanner(System.in);
        char key;
        boolean loop = true;
        while (loop) {
            //根据输入的不同字母,进行对应的操作
            System.out.println("a(add):添加一个数据");
            System.out.println("g(get):取出一个数据");
            System.out.println("h(head):展示头部数据");
            System.out.println("s(show):展示所有数据");
            System.out.println("q(quit):退出程序");
            //因为判满条件的缘故,队列的最大容量实为maxSize-1
            System.out.printf("(队列的最大容量为:%d)\n",maxSize-1);
            key = scanner.next().charAt(0);//接收一个字符
            try {
                switch (key) {
                    case 'a':
                        System.out.println("请输入一个数:");
                        int value = scanner.nextInt();
                        queue.addQueue(value);
                        System.out.println("数据添加成功。");
                        break;
                    case 'g':
                        System.out.printf("取出的数据为:%d\n", queue.getQueue());
                        break;
                    case 'h':
                        queue.headQueue();
                        break;
                    case 's':
                        queue.showQueue();
                        break;
                    case 'q':
                        scanner.close();
                        loop = false;
                        System.out.println("程序已退出。");
                        break;
                    default:
                        break;
                }
            } catch (RuntimeException e) {
                System.out.println(e.getMessage());
            }
        }
    }
}
/**
 * 队列的顺序表示
 * 数组模拟循环队列
 */
class CircleArrayQueue{
    private int maxSize;
    private int front;//指向头元素所在位置
    private int rear;//指向尾元素所在位置的下一个位置
    private int arr[];//存放数据的数组

    //构造器,传入数组最大容量值
    public CircleArrayQueue(int size){
        maxSize = size;
        front = 0;
        rear = 0;
        //虽然数组最大容量为maxSize,但实际用于存储的只有maxSize-1
        arr = new int[maxSize];
    }
    //判空条件:front == rear
    public boolean isEmpty(){
        return front == rear;
    }
    //判满条件:(rear+1)%maxSize == front
    public boolean isFull(){
        return (rear+1)%maxSize == front;
    }
    //添加数据,入队
    public void addQueue(int n){
        //判满
        checkFull();
        arr[rear] = n;
        rear = (rear+1)%maxSize;
    }
    //取出数据,出队
    public int getQueue(){
        //判空
        checkEmpty();
        int value = arr[front];
        front = (front+1)%maxSize;
        return value;
    }
    //队列中的有效值个数
    public int size(){
        return (rear-front+maxSize)%maxSize;
    }
    //展示队列的所有数据
    public void showQueue(){
        //判空
        checkEmpty();
        for(int i=front;i<front+size();i++){
            System.out.printf("arr[%d]=%d\n",i%maxSize,arr[i%maxSize]);
        }
        System.out.printf("当前front=%d,rear=%d\n",front,rear);
    }
    //展示位于队列头部的元素值,不改变front的指向
    public void headQueue(){
        //判空
        checkEmpty();
        System.out.printf("头部数据是:%d\n",arr[front]);
    }
    //检测出队列为空时,打印当前front,rear的值,抛出异常
    public void checkEmpty(){
        if (isEmpty()) {
            System.out.printf("当前front=%d,rear=%d\n",front,rear);
            throw new RuntimeException("队列为空,不能取数据");
        }
    }
    //检测出队列满时,打印当前front,rear的值,抛出异常
    public void checkFull(){
        if(isFull()){
            System.out.printf("当前front=%d,rear=%d\n",front,rear);
            throw new RuntimeException("队列已满,不能添加数据");
        }
    }
}

五、运行结果




到此这篇关于Java基础之数组模拟循环队列的文章就介绍到这了,更多相关Java数组模拟循环队列内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • java二维数组指定不同长度实例方法

    我们知道二维数组,是在一维数组的基础上进行了维度的增加.那么在实际使用的过程中,有时候我们所需要的二维数组,它们其中的维度是不同的,这就需要我们手动的进行设置.下面我们就来一起探究,二维数组在改变维度长度大小的方法,并辅以实例进行代码展示. 1.二维数组的每个元素都是一个一维数组,这些数组不一定都是等长的.声明二维数组的时候可以只指定第一维大小,空缺出第二维大小,之后再指定不同长度的数组.但是注意,第一维大小不能空缺(不能只指定列数不指定行数). public class ArrayTest4

  • 使用java一维数组模拟压栈弹栈

    思路 先进后出,优先解决压栈的问题,之后解决弹栈和main方法 功能 随时模拟压栈 随时模拟弹栈 防止异常和各种错误 随时可以遍历"栈"中存在的变量的方法,压栈弹栈栈帧清晰可见! 使用演示: 压栈: 栈满检测: 遍历栈内存和栈帧: 只要栈中有变量就会输出栈帧: 弹栈: 栈空检测:(没有变量,栈帧不输出!) 源码: import java.util.Scanner; public class MoveTest01 { //局部变量供栈方法的遍历数组使用 static int i; //创

  • java中用数组实现环形队列的示例代码

    本篇文章主要讲述了使用数组实现环形队列的思路以及具体代码 一.队列是什么 我们先来看下百科的解释: 队列是一种特殊的线性表,特殊之处在于它只允许在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作,队列是一种操作受限制的线性表.进行插入操作的端称为队尾,进行删除操作的端称为队头. 总结起来两点: 1.一种线性表 2.添加操作只能在表尾,删除操作在表头(先进先出) 二.实现队列的思路 1.初始化一个空队列 初始化一个大小固定的数组,并将头指针,尾指针都指向下表为0的位置,但其

  • java 定义长度为0的数组/空数组案例

    如下: int[] array = new int[0]; // 定义一个长度为 0 的数组 / 空数组 Sring[] arr = new String[0]; // 定义一个长度为 0 的数组 / 空数组 长度为 0 的数组 / 空数组 并不是 null 有时数组里可能只有一个空字符串 "",这时数组长度是 1.这种情况也要注意判断. if ( arr.length == 1 && arr[ 0 ].equals( "" ) ) { System

  • Java基础之数组详解

    前言 我们了解数组这个概念之前,我们先思考下面几个问题. 如果我们需要两个数据,那么直接创建两个变量即可 int a; int b; 如果需要五个数据,那么可以创建五个变量 int a; int b; int c; int d; int f; 但如果我们需要100个甚至是1万个数据,那么我们创一万个变量?显然这是不现实的.这个时候就需要我们的数组来起作用!帮我们"批量"创建变量. 由上可以得出:数组的本质就是让我们能"批量"创建相同类型的变量! 一.数组的概念 数组

  • java循环练习的简单代码实例

    ★打印九九乘法表 复制代码 代码如下: public class TestDemo {     public static void main(String[] args){         for(int b=1;b<10;b++){             for(int a=1;a<=b;a++)                 System.out.print(a+"*"+b+"="+a*b+"\t");           

  • Java 自定义动态数组方式

    Java自定义动态数组 1.静态数组向动态数组转变 (1)静态数组,数组空间固定长度 这个数组空间总长为4,如果此时新插入一个数据就会报数组空间不足 (2)静态数组如何转变成动态数组 第一步:创建一个空间是data数组两倍的newData数组(扩容): 第二步:把data数组中的元素全部赋值到newData数组: 2.数组扩容程序 // 数组扩容 private void resize(int newCapacity){ E[] newData = (E[]) new Object[newCap

  • 几道java循环练习题(适合新人)

    1.求水仙花数 打印出所有的"水仙花数",所谓"水仙花数"是指一个三位数, 其各位数字立方和等于该数本身. 例如:153是一个"水仙花数", 因为153=1的三次方+5的三次方+3的三次方. //第一题,水仙花数 public class demo4 { public static void main(String[] args) { System.out.println("1000以内的数字"); int u=0; for(

  • Java循环队列原理与用法详解

    本文实例讲述了Java循环队列原理与用法.分享给大家供大家参考,具体如下: 在正式进行循环队列学习之前,我们先来看看在顺序队列中删除队首元素出现的问题 (1)设一个容量为capacity=8,size=5(a,b,c,d,e)的数组,左侧为队首.右侧为队尾. (2)出队一个元素后,需整体往前移动一位 #出队 #2整体前移一位 关于该种操作方式我们很容易得出时间复杂度为O(n). 这时我们就想可不可以在出队元素后,整体元素不往前移,而是在数组中记下队首front是谁,同时队尾tail指向在下一次元

  • Java循环调用多个timer实现定时任务

    通常在使用java实现定时任务时,有两种方法,一种是spring中的schedule(cron = " */5 * * * ?"),另一种就是java中的timer, timer+TimerTask配合实现,这里附上Timer对象的一些常用api Timer() 创建-个新计时器. Timer(boolean isDaemon) 创建一个新计时器, 可以指定其相关的线程作为守护程序运行. Timer(String, name) 创建一个新计时器,其相关的线程具有指定的名称. Timer

随机推荐