热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Java编程:队列

队列:先进先出队列是一个有序列表,可以用数组实现或者链表实现。遵循先入先出的原则。即:先存入队列的数据,要先取出。后存入队

队列:先进先出


  1. 队列是一个有序列表,可以用数组实现或者链表实现。
  2. 遵循先入先出的原则。即:先存入队列的数据,要先取出。后存入队列的数据,要后取出。

顺序队列


思路:


  1. maxSize是该队列的对大容量。front和rear分别记录队列的前后端下标,front会随着数据的输出而改变,rear随着数据输入而改变。
  2. rear指向最后一个数据,添加数据的时候需要先rear++再添加数据。
  3. front指向第一个数据的上一个位置,输出数据的时候需要先front++再输出数据。
  4. 初始:front = -1 rear = -1
  5. 队列为空条件:front = rear
  6. 队列已满条件:rear = maxSize - 1

代码实现:

package queue;import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Scanner;public class ArrayQueueDemo {public static void main(String[] args) {// 创建一个队列ArrayQueue arrayQueue = new ArrayQueue(3);char key = ' '; // 接收用户输入Scanner scanner = new Scanner(System.in); // 接收输入boolean loop = true;while (loop) {System.out.println("s(show):显示队列");System.out.println("e(quit):退出程序");System.out.println("a(add):添加数据");System.out.println("g(get):取出数据");System.out.println("h(head):查看队列头数据");key = scanner.next().charAt(0); // 接收一个字符switch (key) {case 's': // 显示队列arrayQueue.showQueue();break;case 'e': // 退出scanner.close();loop = false;break;case 'a': // 添加一个数据System.out.println("请输入一个数据:");int value = scanner.nextInt();arrayQueue.addQueue(value);break;case 'g': // 取出一个数据try {int get = arrayQueue.getQueue();System.out.printf("取出的数据为:%d", get);} catch (Exception e) {System.out.println(e.getMessage());}break;case 'h': // 取出头数据try {int get = arrayQueue.headQueue();System.out.printf("取出的头数据为:%d", get);} catch (Exception e) {System.out.println(e.getMessage());}break;default:break;}}System.out.println("程序已退出");}}// 使用数组模拟队列——使用数组编写一个ArrayQueue类
class ArrayQueue {private final int maxSize; // 表示数组最大容量private int front; // 队列头private int rear; // 队列尾private int[] arr; // 该数组用于存放数据,模拟队列// 创建队列构造器public ArrayQueue(int arrMaxSize) {this.maxSize = arrMaxSize;this.arr = new int[maxSize];this.front = -1; // 指向队列头部,分析出front指向队列头的前一个位置this.rear = -1; // 指向队列尾,分析出rear指向队列尾的数据(即就是队列最后一个数据)}// 判断队列是否已满public boolean isFull() {// maxSize是数组长度return this.rear == this.maxSize - 1;}// 判断队列是否为空public boolean isEmpty() {return this.rear == this.front;}// 添加数据到队列public void addQueue(int data) {if (this.isFull()) {System.out.println("队列已满,无法添加新数据!");return;}this.arr[++this.rear] = data; // 先让rear后移,然后存入数据}// 数据出队列public int getQueue() {if (this.isEmpty()) {// 抛出异常throw new RuntimeException("队列为空,无法取出数据!");}return this.arr[++this.front];}// 显示队列数据public void showQueue() {if (isEmpty()) {System.out.println("队列为空!");return;}
// for (int n : this.arr){
// System.out.printf("%d ",n);
// }for (int i &#61; 0; i < this.arr.length; i&#43;&#43;) {System.out.printf("arr[%d]:%d\n", i, this.arr[i]);}}// 显示队列头数据public int headQueue() {// 判断if (isEmpty()) {throw new RuntimeException("数列为空&#xff01;");}return this.arr[front &#43; 1];}// 显示队列长度public int queueLen() {return this.arr.length;}
}

问题&#xff1a;


  1. 目前数组使用一次就无法使用&#xff0c;没有达到复用的效果。
  2. 将这个数组使用算法&#xff0c;改进成环形数组。核心%取模

数组模拟环形队列


思路如下&#xff1a;


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

代码实现&#xff1a;

package queue;import java.util.Scanner;public class CircleArrayQueueDemo {// 测试public static void main(String[] args) {// 创建一个队列CircleArrayQueue arrayQueue &#61; new CircleArrayQueue(3);char key &#61; &#39; &#39;; // 接收用户输入Scanner scanner &#61; new Scanner(System.in); // 接收输入boolean loop &#61; true;while (loop) {System.out.println("s(show)&#xff1a;显示队列");System.out.println("e(quit)&#xff1a;退出程序");System.out.println("a(add)&#xff1a;添加数据");System.out.println("g(get)&#xff1a;取出数据");System.out.println("h(head)&#xff1a;查看队列头数据");key &#61; scanner.next().charAt(0); // 接收一个字符switch (key) {case &#39;s&#39;: // 显示队列arrayQueue.showQueue();break;case &#39;e&#39;: // 退出scanner.close();loop &#61; false;break;case &#39;a&#39;: // 添加一个数据System.out.println("请输入一个数据&#xff1a;");int value &#61; scanner.nextInt();arrayQueue.addQueue(value);break;case &#39;g&#39;: // 取出一个数据try {int get &#61; arrayQueue.getQueue();System.out.printf("取出的数据为&#xff1a;%d", get);} catch (Exception e) {System.out.println(e.getMessage());}break;case &#39;h&#39;: // 取出头数据try {int get &#61; arrayQueue.headQueue();System.out.printf("取出的头数据为&#xff1a;%d", get);} catch (Exception e) {System.out.println(e.getMessage());}break;default:break;}}System.out.println("程序已退出");}
}// 使用数组模拟队列——使用数组编写一个ArrayQueue类
class CircleArrayQueue {private final int maxSize; // 表示数组最大容量private int front; // 队列头&#xff1a;指向队列头private int rear; // 队列尾&#xff1a;指向队列尾下一个位置private int[] arr; // 该数组用于存放数据&#xff0c;模拟队列// 创建队列构造器public CircleArrayQueue(int arrMaxSize) {this.maxSize &#61; arrMaxSize;this.arr &#61; new int[maxSize];}// 判断队列是否已满public boolean isFull() {// maxSize是数组长度return (this.rear &#43; 1) % this.maxSize &#61;&#61; this.front;}// 判断队列是否为空public boolean isEmpty() {return this.rear &#61;&#61; this.front;}// 添加数据到队列public void addQueue(int data) {if (this.isFull()) {System.out.println("队列已满&#xff0c;无法添加新数据&#xff01;");return;}this.arr[this.rear] &#61; data; // 先让rear后移&#xff0c;然后存入数据this.rear &#61; (this.rear &#43; 1) % this.maxSize; // rear后移&#xff0c;考虑取模}// 数据出队列public int getQueue() {if (this.isEmpty()) {// 抛出异常throw new RuntimeException("队列为空&#xff0c;无法取出数据&#xff01;");}// 这里需要分析出front是指向队列第一个元素// 1. 先把front对应的值保存在一个临时变量// 2. 将front后移// 3. 将临时保存的变量返回int value &#61; arr[this.front];this.front &#61; (this.front &#43; 1) % this.maxSize;return value;}// 显示队列数据public void showQueue() {if (isEmpty()) {System.out.println("队列为空&#xff01;");return;}// 从front开始遍历&#xff0c;遍历多少个元素for (int i &#61; this.front; i < this.front &#43; size(); i&#43;&#43;) {System.out.printf("arr[%d]&#61;%d\n", i % this.maxSize, arr[i % this.maxSize]);}}// 求出当前队列有效数据个数public int size() {return (this.rear &#43; this.maxSize - this.front) % this.maxSize;}// 显示队列头数据public int headQueue() {// 判断if (isEmpty()) {throw new RuntimeException("数列为空&#xff01;");}return this.arr[this.front];}// 显示队列长度public int queueLen() {return this.arr.length;}
}

推荐阅读
  • 本文介绍了如何在给定的有序字符序列中插入新字符,并保持序列的有序性。通过示例代码演示了插入过程,以及插入后的字符序列。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了在Java中gt、gtgt、gtgtgt和lt之间的区别。通过解释符号的含义和使用例子,帮助读者理解这些符号在二进制表示和移位操作中的作用。同时,文章还提到了负数的补码表示和移位操作的限制。 ... [详细]
  • 本文介绍了基于c语言的mcs51单片机定时器计数器的应用教程,包括定时器的设置和计数方法,以及中断函数的使用。同时介绍了定时器应用的举例,包括定时器中断函数的编写和频率值的计算方法。主函数中设置了T0模式和T1计数的初值,并开启了T0和T1的中断,最后启动了CPU中断。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 本文介绍了Java高并发程序设计中线程安全的概念与synchronized关键字的使用。通过一个计数器的例子,演示了多线程同时对变量进行累加操作时可能出现的问题。最终值会小于预期的原因是因为两个线程同时对变量进行写入时,其中一个线程的结果会覆盖另一个线程的结果。为了解决这个问题,可以使用synchronized关键字来保证线程安全。 ... [详细]
  • 本文探讨了C语言中指针的应用与价值,指针在C语言中具有灵活性和可变性,通过指针可以操作系统内存和控制外部I/O端口。文章介绍了指针变量和指针的指向变量的含义和用法,以及判断变量数据类型和指向变量或成员变量的类型的方法。还讨论了指针访问数组元素和下标法数组元素的等价关系,以及指针作为函数参数可以改变主调函数变量的值的特点。此外,文章还提到了指针在动态存储分配、链表创建和相关操作中的应用,以及类成员指针与外部变量的区分方法。通过本文的阐述,读者可以更好地理解和应用C语言中的指针。 ... [详细]
  • 个人学习使用:谨慎参考1Client类importcom.thoughtworks.gauge.Step;importcom.thoughtworks.gauge.T ... [详细]
author-avatar
那一抹细雨
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有