java数据结构,一个案例带你用数组模拟队列,环形队列!

简介: java数据结构,一个案例带你用数组模拟队列,环形队列!

队列

  • 队列是一个有序列表,可以用数组(顺序存储)或是链表(链式存储)来实现。
  • 遵循先入先出的原则。即:先存入队列的数据,要先取出。后存入的要后取出。

20210614175031206.png

使用数组模拟队列

  • 队列本身是有序列表,若使用数组的结构来存储队列的数据,则队列数组的声明如下图,其中maxSize是该队列的最大容量。
  • 因为队列的输出,输入是分别从前后端来处理,因此需要两个变量front及rear分别记录队列前后端的下标,front会随着数据输出从而改变,而rear则是随着数据输入而改变。

如图所示:

20210614175709508.png

思路分析:

当我们将数据存入队列时称为 addQueueaddQueue 的处理需要有两个步骤

  1. 将尾指针往后移:rear+1,当front == rear[空]
  2. 若尾指针 rear 小于队列的最大下标 maxSize - 1,则将数据存入 rear 所指的数组元素中,否则无法存入数据。rear == maxSize -1[队列满]

代码实现:

使用数组模拟队列,编写一个ArrayQueue类

   class ArrayQueue{
        private int maxSize; //表示数组的最大容量
        private int front; //队列头
        private int rear; //队列尾
        private int [] arr; //该数组用于存放数据,模拟队列
        //创建队列的构造器
        public ArrayQueue(int arrMaxSize){
            this.maxSize=arrMaxSize;
            this.arr=new int[this.maxSize];
            this.front=-1; //指向队列头部,分析出front是指向队列头的前一个位置。
            this.rear=-1; //指向队列尾,指向队列尾的数据(就是队列中最后一个数据)。
        }
        //判断队列是否满
        public boolean isFull(){
            return this.rear == this.maxSize-1;
        }
        //判断队列是否为空
        public boolean isEmpty(){
            return this.rear == this.front;
        }
        //添加数据到队列
        public void addQueue(int n){
            if (isFull()){
                System.out.println("队列满,不能加入数据");
                return;
            }
            this.rear++; //让rear后移
            this.arr[rear]=n;
        }
        //获取队列的数据,出队列
        public int getQueue(){
            //判断队列是否为空
            if (isEmpty()){
                //通过抛出异常
                throw new RuntimeException("队列空,不能取数据");
            }
            front++; //front后移
            return arr[front];
        }
        //显示队列的所有数据
        public void showQueue(){
            if (isEmpty()){
                System.out.println("队列空,没有数据");
            }
            for (int i=0;i<arr.length;i++){
                System.out.printf("arr[%d]=%d\n",i,arr[i]);
            }
        }
        //显示队列的头数据,注意:不是取数据
        public int headQueue(){
            //判断
            if (isEmpty()){
                throw new RuntimeException("队列空的,没有数据");
            }
            return arr[front+1];
        }
    }

对模拟队列进行测试

    public static void main(String[] args) {
        ArrayQueue arrayQueue= new ArrayQueue(3);
        Scanner inpunt =new Scanner(System.in);
        Boolean b=true;
        do {
            switch (inpunt.nextInt()){
                case 1:{
                    try {
                        System.out.println("显示队列所有数据");
                        arrayQueue.showQueue();
                    }catch (Exception e){
                        e.printStackTrace();
                    }
                    break;
                }
                case 2:{
                    System.out.println("添加数据到队列");
                    arrayQueue.addQueue(inpunt.nextInt());
                    break;
                }
                case 3:{
                    try {
                        System.out.println("显示队列的头数据");
                        int headQueue = arrayQueue.headQueue();
                        System.out.println("队列头:"+headQueue);
                    }catch (Exception e){
                        e.printStackTrace();
                    }
                    break;
                }
                case 4:{
                    try {
                        System.out.println("从队列中取数据");
                        int queueQueue = arrayQueue.getQueue();
                        System.out.println("取数据:"+queueQueue);
                    }catch (Exception e){
                        e.printStackTrace();
                    }
                    break;
                }
                default:{
                    b=false;
                }
            }
        }while (b);
    }
  • 测试1:添加数据并显示队列所有数据 可以看出现在队列中添加了三个数据分别为 10,20,30

20210620232220950.png

测试2:查看头数据,然后取出数据后,再次查看头数据 因为队列遵循先进先出的原则,这边取完数据再查看队列,头数据为第二次添加的数据。

2021062023253819.png


测试3:取出所有数据在添加数据。注意看这里,当我们数据取完后,在添加数据就会出现问题。


2021062023293832.png

问题分析及优化:

  • 目前我们的数组使用一次就不能用了,没有达到复用的效果。
  • 我们可以将这个数组使用算法改进成一个环形的队列。

使用数组模拟环形队列

20210626162533225.png


思路分析

  1. front变量的含义做一个调整,front就指向队列的第一个元素,也就是说 arr[front]就是队列的第一个元素,front的初始值 = 0;
  2. rear 变量的含义也做一个调整,rear指向队列的最后一个元素的最后一个位置。因为希望空出一个空间做为约定。rear的初始值也 = 0;
  3. 当队列满时,条件是 (rear+1)% maxSize = front [满]
  4. 当队列为空的条件是rear == front空。
  5. 队列中有效数据的个数 (rear+maxSize-front)%maxSize // rear = 1 front = 0

代码实现

对前面的数组模拟队列进行优化,充分利用数组,因此将数组看作是一个环形的。(通过取模的方式来实现即可)

class CircleArray{
    private int maxSize; //表示数组的最大容量
    //front变量的含义做一个调整,front就指向队列的第一个元素,也就是说 arr[front]就是队列的第一个元素,front的初始值 = 0;
    private int front;
    //rear 变量的含义也做一个调整,rear指向队列的最后一个元素的最后一个位置。因为希望空出一个空间做为约定。rear的初始值也 = 0;
    private int rear;
    private int [] arr; //该数组用于存放数据,模拟队列
    public CircleArray(int arrMaxSize){
        maxSize = arrMaxSize;
        arr = new int[maxSize];
    }
    //判断队列是否满
    public boolean isFull(){
        return (rear + 1) % maxSize == front;
    }
    //判断队列是否为空
    public boolean isEmpty(){
        return this.rear == this.front;
    }
    //添加数据到队列
    public void addQueue(int n){
        if (isFull()){
            System.out.println("队列满,不能加入数据");
            return;
        }
        //直接将数据加入
        arr[rear] = n;
        //将 rear 后移,这里必须考虑去模
        rear = (rear + 1) % maxSize;
    }
    //获取队列的数据,出队列
    public int getQueue(){
        //判断队列是否为空
        if (isEmpty()){
            //通过抛出异常
            throw new RuntimeException("队列空,不能取数据");
        }
        /**
         *   这里需要分析出 front 是指向队列的第一个元素
         *   1.先把 front 对应的值保留到一个临时变量
         *   2.将 front 后移,考虑取模
         *   3.将临时保存的变量返回
         */
        int value = arr[front];
        front = (front + 1) % maxSize;
        return value;
    }
    //显示队列的所有数据
    public void showQueue(){
        //遍历
        if (isEmpty()){
            throw  new RuntimeException("队列空,没有数据");
        }
        //思路:从front开始遍历,遍历多少个元素
        for (int i= front; i < front + size();i++){
            System.out.printf("arr[%d]=%d\n",i % maxSize,arr[i % maxSize]);
        }
    }
    //求出当前队列有效数据的个数
    public int size(){
        return (rear + maxSize - front) % maxSize;
    }
    //显示队列的头数据,注意:不是取数据
    public int headQueue(){
        //判断
        if (isEmpty()){
            throw new RuntimeException("队列空的,没有数据");
        }
        return arr[front];
    }
}

对环形队列进行测试

    public static void main(String[] args) {
        //测试数组模拟环形队列
        CircleArray circleArray= new CircleArray(4);  //设置4,其队列的有效数据最大是3
        Scanner inpunt =new Scanner(System.in);
        Boolean b=true;
        do {
            switch (inpunt.nextInt()){
                case 1:{
                    try {
                        System.out.println("显示队列所有数据");
                        circleArray.showQueue();
                    }catch (Exception e){
                        e.printStackTrace();
                    }
                    break;
                }
                case 2:{
                    System.out.println("添加数据到队列");
                    circleArray.addQueue(inpunt.nextInt());
                    break;
                }
                case 3:{
                    try {
                        System.out.println("显示队列的头数据");
                        int headQueue = circleArray.headQueue();
                        System.out.println("队列头:"+headQueue);
                    }catch (Exception e){
                        e.printStackTrace();
                    }
                    break;
                }
                case 4:{
                    try {
                        System.out.println("从队列中取数据");
                        int queueQueue = circleArray.getQueue();
                        System.out.println("取数据:"+queueQueue);
                    }catch (Exception e){
                        e.printStackTrace();
                    }
                    break;
                }
                default:{
                    b=false;
                }
            }
        }while (b);
    }
  • 测试1往环形队列中添加数据
    20210703152630365.png
  • 测试2 从环形队列中取数据
    20210703153035186.png
  • 测试3 取数据后重新添加数据
    20210703153101911.png

相关文章
|
2月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
230 9
|
3月前
|
存储 Java
Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。
【10月更文挑战第19天】本文详细介绍了Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。HashMap以其高效的插入、查找和删除操作著称,而TreeMap则擅长于保持元素的自然排序或自定义排序,两者各具优势,适用于不同的开发场景。
55 1
|
3月前
|
存储 Java
告别混乱!用Java Map优雅管理你的数据结构
【10月更文挑战第17天】在软件开发中,随着项目复杂度增加,数据结构的组织和管理至关重要。Java中的Map接口提供了一种优雅的解决方案,帮助我们高效、清晰地管理数据。本文通过在线购物平台的案例,展示了Map在商品管理、用户管理和订单管理中的具体应用,有效提升了代码质量和维护性。
97 2
|
3月前
|
存储 Java 开发者
Java Map实战:用HashMap和TreeMap轻松解决复杂数据结构问题!
【10月更文挑战第17天】本文深入探讨了Java中HashMap和TreeMap两种Map类型的特性和应用场景。HashMap基于哈希表实现,支持高效的数据操作且允许键值为null;TreeMap基于红黑树实现,支持自然排序或自定义排序,确保元素有序。文章通过具体示例展示了两者的实战应用,帮助开发者根据实际需求选择合适的数据结构,提高开发效率。
85 2
|
20天前
|
存储 缓存 安全
Java 集合江湖:底层数据结构的大揭秘!
小米是一位热爱技术分享的程序员,本文详细解析了Java面试中常见的List、Set、Map的区别。不仅介绍了它们的基本特性和实现类,还深入探讨了各自的使用场景和面试技巧,帮助读者更好地理解和应对相关问题。
37 5
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
68 5
|
2月前
|
缓存 算法 Java
本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制
在现代软件开发中,性能优化至关重要。本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制。通过调整垃圾回收器参数、优化堆大小与布局、使用对象池和缓存技术,开发者可显著提升应用性能和稳定性。
52 6
|
2月前
|
存储 Java 索引
Java中的数据结构:ArrayList和LinkedList的比较
【10月更文挑战第28天】在Java编程世界中,数据结构是构建复杂程序的基石。本文将深入探讨两种常用的数据结构:ArrayList和LinkedList,通过直观的比喻和实例分析,揭示它们各自的优势与局限,帮助你在面对不同的编程挑战时做出明智的选择。
|
2月前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
3月前
|
存储 算法 Java
Java 中常用的数据结构
【10月更文挑战第20天】这些数据结构在 Java 编程中都有着广泛的应用,掌握它们的特点和用法对于提高编程能力和解决实际问题非常重要。
35 6