资讯专栏INFORMATION COLUMN

Java数据结构与算法[原创]——队列

韩冰 / 2172人阅读

摘要:前言数据结构与算法专题会不定时更新,欢迎各位读者监督。队列和栈类似,也是一个遵循特殊规则约束的数据结构。将没有元素的队列称之为空队,往队列中插入元素的过程称之为入队,从队列中移除元素的过程称之为出队。

</>复制代码

  1. 声明:码字不易,转载请注明出处,欢迎文章下方讨论交流。

前言:Java数据结构与算法专题会不定时更新,欢迎各位读者监督。本文介绍数据结构中的队列(queue)的概念、存储结构、队列的特点,文末给出java实现循环队列的代码实现供读者参考学习。

1.队列的概念

队列正如其名,队列就像一支队伍,有队首(head)队尾(tail)以及队列长度。队列和栈类似,也是一个遵循特殊规则约束的数据结构。在上一篇文章java数据结构与算法——栈中介绍栈是一个后进先出(LIFO)的数据结构,队列正好与之相反,是一个先进先出(FIFO,First In First Out),例如我们去肯德基排队,先排上队的肯定先拿到餐出队,这和我们对列认知是一致的。

上面说到队列是一个遵循特殊规则的数据结构,除了先进先出,队列的插入只能从队列的一端操作,我们称这端为队尾;对应的,移除只能从另一端出来,我们称之为队首。

将没有元素的队列称之为空队,往队列中插入元素的过程称之为入队,从队列中移除元素的过程称之为出队。

一般而言,队列的实现有两种方式:数组实现和链表实现,本篇中采取数组实现,链表实现在后续补充。用数组实现的队列有两种:一种是顺序队列,另一种是循环队列,这两种队列的存储结构和特点下文会逐一介绍。

说明:用数组实现队列,若队列中出现队满的情况(因为在声明队列时,一般会指定一个初始容量),此时如果有新元素入队,但没有位置怎么办?要么丢弃,要么等待。

2.队列的存储结构

以下采用数组实现,初始化一个队列长度为6,队列有两个标记,一个队头的位置head,一个队尾的位置tail,初始都指向数组下标为0的位置,如图所示:

在插入元素时,tail标记+1,比如入队三个元素,依次为A,B,C(注意是有顺序的),则当前队列存储情况如图:
当前head为0,tail为3,接下来进行出队操作,此处将A元素出队,则head+1,此时队列的存储情况如图:

根据上面的图例,我们可以通过tail-head获得队列中元素的个数。当tail==head时,此时队列为空,而当tail等于数组长度时,此时将无法出入元素,那么队列是否已经满呢?未必!因为head和tail在入队和出队操作中只增不减,因此head和tail都会最终指向队列之外的存储位置,此时虽然数组为空,但也无法将元素入队。

上溢: 当队列中无法插入元素时,称之为上溢;
假上溢: 在顺序队列中,数组还有空间(不一定全空)但无法入队称之为假上溢;
真上溢: 如果head为0,tail指向数组之外,即数组真满了,称之为真上溢;
下溢: 如果空队中执行出队操作,此时队列中无元素,称之为下溢

如何解决“假上溢”的问题呢?此时引入循环队列。出现假上溢时,此时数组还有空闲的位置,将tail从新指向数组的0索引处即可,如图所示:

如果继续入队E和F,则队列的存储结构如图:
通常而言,在对head和tail加1时,为了方便采用对数组长度取余操作。另外由于顺序队列存在“假上溢”的现象,所以一般用循环队列实现。

在采用循环队列实现的过程中,当队列满队时,tail等于head,而当队列空时,tail也等于head,为了区分两种状态,一般规定循环队列的长度为数组长度-1,即有一个位置不放元素,此时head==tail时为空队,而当head==(tail+1)%数组长度,说明对满。

3.队列的java代码实现

下面是循环队列的java代码,采用数组方式实现:

</>复制代码

  1. public class ArrayQueue {
  2. private final Object[] queue; //声明一个数组
  3. private int head;
  4. private int tail;
  5. /**
  6. * 初始化队列
  7. * @param capacity 队列长度
  8. */
  9. public ArrayQueue(int capacity){
  10. this.queue = new Object[capacity];
  11. }
  12. /**
  13. * 入队
  14. * @param o 入队元素
  15. * @return 入队成功与否
  16. */
  17. public boolean put(Object o){
  18. if(head == (tail+1)%queue.length){
  19. //说明队满
  20. return false;
  21. }
  22. queue[tail] = o;
  23. tail = (tail+1)%queue.length; //tail标记后移一位
  24. return true;
  25. }
  26. /**
  27. * 返回队首元素,但不出队
  28. * @return
  29. */
  30. public Object peak() {
  31. if(head==tail){
  32. //队空
  33. return null;
  34. }
  35. return queue[head];
  36. }
  37. /**
  38. * 出队
  39. * @return 出队元素
  40. */
  41. public Object pull(){
  42. if(head==tail){
  43. return null;
  44. }
  45. Object item = queue[head];
  46. queue[head] = null;
  47. return item;
  48. }
  49. /**
  50. * 判断是否为空
  51. * @return
  52. */
  53. public boolean isEmpty(){
  54. return head == tail;
  55. }
  56. /**
  57. * 判断是否为满
  58. * @return
  59. */
  60. public boolean isFull(){
  61. return head == (tail+1)%queue.length;
  62. }
  63. /**
  64. * 获取队列中的元素个数
  65. * @return
  66. */
  67. public int getsize(){
  68. if(tail>=head){
  69. return tail-head;
  70. }else{
  71. return (tail+queue.length)-head;
  72. }
  73. }
  74. }

下面是队列的测试代码

</>复制代码

  1. public class ArrayQueueTest {
  2. public static void main(String[] args) {
  3. ArrayQueue q = new ArrayQueue(4); //初始化队列长度为3,因为循环队列有一个不放元素
  4. System.out.println(q.put("张三")); //入队,true
  5. System.out.println(q.put("李四")); //入队,true
  6. System.out.println(q.put("赵五")); //入队,true
  7. System.out.println(q.put("老王")); //队满,false
  8. System.out.println(q.isFull()); //队满 true
  9. System.out.println(q.getsize()); //3,队列中有3个元素
  10. System.out.println(q.peak()); //返回“张三” 不出队
  11. System.out.println(q.pull()); //返回“张三”
  12. System.out.println(q.pull()); //返回“李四”
  13. System.out.println(q.pull()); //返回“赵五”
  14. System.out.println(q.isEmpty()); //true 空队
  15. }
  16. }
4.队列的应用场景

队列先入先出的特点,使得其应用非常广泛,比如队列作为“缓冲区”,可以解决计算机和外设速度不匹配的问题,FIFO的特点保证了数据传输的顺序;除此之外队列在后面树的层序遍历中也有应用,FIFO的特点保证了处理顺序不会出错。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/76372.html

相关文章

  • CAS 算法 —— Compare and Swap

    摘要:算法算法会先对一个内存变量位置和一个给定的值进行比较,如果相等,则用一个新值去修改这个内存变量位置。因为是非公平锁,所以一上来就尝试抢占锁给定旧值并希望用新值去更新内存变量。 本文翻译和原创各占一半,所以还是厚颜无耻归类到原创好了...https://howtodoinjava.com/jav...java 5 其中一个令人振奋的改进是新增了支持原子操作的类型,例如 AtomicInt...

    mmy123456 评论0 收藏0
  • 长知识 - 收藏集 - 掘金

    摘要:问题是这些服务都是第三方提供的,不能保证它们的响应时间,快的话美团点评分布式生成系统后端掘金背景在复杂分布式系统中,往往需要对大量的数据和消息进行唯一标识。 SpringBatch 读取 txt 文件并写入数据库 - 后端 - 掘金SpringBatch 读取 txt 文件并写入数据库... Java 进阶-多线程开发关键技术 - 后端 - 掘金原创文章,转载请务必将下面这段话置于文章...

    SimpleTriangle 评论0 收藏0
  • 长知识系列 - 收藏集 - 掘金

    摘要:问题是这些服务都是第三方提供的,不能保证它们的响应时间,快的话美团点评分布式生成系统后端掘金背景在复杂分布式系统中,往往需要对大量的数据和消息进行唯一标识。 SpringBatch 读取 txt 文件并写入数据库 - 后端 - 掘金SpringBatch 读取 txt 文件并写入数据库... Java 进阶-多线程开发关键技术 - 后端 - 掘金原创文章,转载请务必将下面这段话置于文章...

    longmon 评论0 收藏0
  • JavaSE数据结构基础知识系列——专栏导航

    ⭐️前面的话⭐️ 大家好!这是Java基础知识与数据结构博文的导航帖,收藏我!学习Java不迷路! ?博客主页:未见花闻的博客主页 ?欢迎关注?点赞?收藏⭐️留言? ?本文由未见花闻原创,CSDN首发! ?首发时间:?2021年11月11日? ✉️坚持和努力一定能换来诗与远方! ?参考书籍:?《Java核心技术卷1》,?《Java核心技术卷2》,?《Java编程思想》 ?参考在线编程网站:?牛...

    Cc_2011 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<