java环形队列_数组实现环形队列Java

article/2025/9/28 1:08:09

用数组实现环形队列的特点是高效。

能快速判断队列是否 满/空;

能快速存取数据。

因为简单高效,所以甚至在硬件中都实现了环形队列。

环形队列广泛应用于网络数据的收发,和不同应用间数据交换(内核和应用程序大量交换数据,从硬件接受大量数据)

内存上没有环形结构,因此环形队列实际上用数组的线性空间来实现。

但是当数据到了尾部如何处理呢?它将转回到0位置来处理。这个转回是通过取模操作来执行的。

因此,环形队列是逻辑上将数组元素q[0]和q[maxSize - 1]连接,形成一个存放队列的环形空间。

为了方便读写,还要用数组下标来指明队列的读写位置。head/tail.其中head指向可以读的位置,tail指向可以写的位置。

c0c44d093ce176285373eb3ee420bc67.png

环形队列的关键是判断队列为空,还是为满。当tail追上head时,队列为满时,当head追上tail时,队列为空。但如何知道谁追上谁。还需要一些辅助的手段来判断.

如何判断环形队列为空,为满有两种判断方法。

一.附加一个标志位tag

当head赶上tail,队列空,则令tag=0,

当tail赶上head,队列满,则令tag=1,

二.限制tail赶上head,即队尾结点与队首结点之间至少留有一个元素的空间。

队列空:   head==tail

队列满:   (tail+1)% MAXN ==head

package ds.queue;

/**

* 用数组实现环形队列

* 附加标志位法

*/

public class ArrayCircleQueue {

private int[] arr;

private int maxSize;

private int rear;

private int front;

// 当front赶上rear,说明队列满,令flag=0;

// 当rear赶上front,说明队列空,令flag=1

private int flag;

public ArrayCircleQueue(int size) {

maxSize = size;

arr = new int[maxSize];

front = 0;

rear = 0;

flag = 1;

}

public boolean isFull() {

return front == rear && flag == 0;

}

public boolean isEmpty() {

return front == rear && flag == 1;

}

public int peek() {

if (isEmpty()) throw new RuntimeException("queue is empty!!!");

return arr[front];

}

public void offer(int v) {

if (isFull()) {

System.out.println("queue is full, could not add.");

return;

}

arr[rear] = v;

rear = (rear + 1) % maxSize;

if (rear == front) flag = 0;

}

public int poll() {

if (isEmpty()) throw new RuntimeException("queue is empty!");

int res = arr[front];

front = (front + 1) % maxSize;

if (front == rear) flag = 1;

return res;

}

public void show() {

if (isEmpty()) {

System.out.println("The queue is empty!!!");

return;

}

System.out.println("front -> rear");

for (int i = front; i < front + size(); i++) {

System.out.print(arr[i % maxSize] + " ");

}

System.out.println();

}

public int size() {

if (rear > front) return rear - front;

return rear + maxSize - front;

}

}

package ds.queue;

/**

* 数组实现环形队列

* -预留空间法

*/

public class ArrayCircleQueue2 {

private int maxSize;

private int front; // 指向队列的第一个元素,arr[front]就是队列的第一个元素。初始值0

private int rear; // 指向队列的最后一个元素的后一个位置,希望空出一个空间作为约定。初始值0

private int[] arr;

public ArrayCircleQueue2(int size) {

maxSize = size;

arr = new int[size];

}

public boolean isFull() {

return (rear + 1) % maxSize == front;

}

public boolean isEmpty() {

return front == rear;

}

public void offer(int v) {

if (isFull()) {

System.out.println("queue is full!!!");

return;

}

arr[rear] = v;

rear = (rear + 1) % maxSize;

}

public int poll() {

if (isEmpty()) throw new RuntimeException("queue is empty!!!");

int res = arr[front];

front = (front + 1) % maxSize;

return res;

}

public int size() {

return (rear + maxSize - front) % maxSize;

}

public int peek() {

if (isEmpty()) throw new RuntimeException("queue is empty");

return arr[front];

}

public void show() {

if (isEmpty()) {

System.out.println("queue is empty!!!");

return;

}

System.out.println("Front -> rear");

for (int i = front; i < front + size(); i++) {

System.out.print(arr[i % maxSize]);

}

System.out.println();

}

}

测试代码

package ds.queue;

import java.util.Scanner;

/**

*/

public class QueueDemo {

public static void main(String[] args) {

ArrayCircleQueue q = new ArrayCircleQueue(5);

char c;

Scanner scanner = new Scanner(System.in);

boolean loop = true;

while (loop) {

System.out.println("s: 显示队列");

System.out.println("e: 退出应用");

System.out.println("a: 元素入队尾");

System.out.println("g: 从队头取出元素");

System.out.println("h: 查看队头元素");

c = scanner.next().charAt(0);

switch (c) {

case 's':

q.show();

break;

case 'a':

System.out.println("请输入一个数");

int v = scanner.nextInt();

q.offer(v);

break;

case 'g':

try {

int res = q.poll();

System.out.println("取出了 " + res);

} catch (Exception e) {

System.out.println(e.getMessage());

}

break;

case 'h':

try {

int res = q.peek();

System.out.println("队头元素 " + res);

} catch (Exception e) {

System.out.println(e.getMessage());

}

break;

case 'e':

loop = false;

break;

default:

break;

}

}

}

}


http://chatgpt.dhexx.cn/article/hMTtttgP.shtml

相关文章

环形队列初步探讨

文章目录 前言一、环形队列二、环形队列基本操作二、示例总结 前言 最近使用队列的时候&#xff0c;在实现大数据整体平移的时候还是用了内存copy虽然暂时达到性能要求&#xff0c;但是总感觉很笨重&#xff0c;最后用环形队列重构了部分代码&#xff0c;效果还行。 一、环形队…

环形队列

介绍 环形队列是队列的一种特殊情况&#xff0c;也是基于队列的实现&#xff0c;队列是动态的集合&#xff0c;而环形队列则是固定长度的&#xff0c;当队列满时&#xff0c;则从队首删除元素。其原理基本和队列一致&#xff0c;都是实现先进先出的策略。 实现 先定义数据&…

C语言,环形队列

什么是环形队列&#xff1f; 环形缓冲区是一个非常典型的数据结构&#xff0c;这种数据结构符合生产者&#xff0c;消费者模型&#xff0c;可以理解它是一个水坑&#xff0c;生产者不断的往里面灌水&#xff0c;消费者就不断的从里面取出水。 那就可能会有人问&#xff0c;既然…

数据结构(10)---队列之环形队列

环形队列 文章目录 环形队列什么是环形队列循环队列的实现第一种实现第二种实现 什么是环形队列 环形队列也是队列的一种数据结构, 也是在队头出队, 队尾入队; 只是环形队列的大小是确定的, 不能进行一个长度的增加, 当你把一个环形队列创建好之后, 它能存放的元素个数是确定的…

数据结构--环形队列的介绍与实现

数据结构--环形队列实现 一、环形队列实现原理环形队列的几个判断条件 二、代码实现1.环形队列类&#xff08;CircleQueue&#xff09;2.环形队列类测试类3.程序运行结果4.完整代码 环形队列可以用数组实现&#xff0c;也可以使用循环链表实现.在使用数组实现循环队列的时候&am…

【数据结构(C语言描述)】环形队列

目录 一、基础知识二、数组实现环队2.1 初始化2.2 判断环队是否为空2.3 判断环队是否为满2.4 入队2.5 出队2.6 取队头元素2.7 取队尾元素2.8 销毁环队 三、链表实现环队3.1 初始化3.2 判断环队是否为空3.3 判断环队是否为满3.4 入队3.5 出队3.6 取队头元素3.7 取队尾元素3.8 销…

环形队列原理,全网最通俗易懂

队列是什么 队列是一种很常见的数据结构&#xff0c;满足先进先出的方式&#xff0c;如果我们设定队列的最大长度&#xff0c;那就意味着进队列和出队列的元素的数量实则满足一种动态平衡。 如果我们把首次添加入队列的元素作为一个一维坐标的原点&#xff0c;那么随着队列中…

Mysql 死锁和死锁的解决方案

最近在研究Mysql底层原理&#xff0c;研究到了死锁&#xff0c;感觉挺有意思&#xff0c;在这里和大家分享一下 前置知识&#xff1a;需要了解锁的种类&#xff0c;如表锁、行锁&#xff1b;行锁又分为记录锁、间隙锁、临键锁等等&#xff1b;什么情况下会加表锁&#xff0c;什…

mysql死锁演示

背景&#xff1a; 线上日志突然爆了有数据库死锁的日志。 通过以下语句查询数据库死锁的日志 SHOW ENGINE INNODB STATUS 通过 日志分析&#xff0c;看到了两条update语句并且是里面有子查询。还有两个表的更新顺序问题。 解决方案是&#xff1a;加了分布式锁&#xff0c;…

记录一次mysql死锁

一&#xff0c;死锁发现 项目中有一个接口包含更新操作1&#xff0c;后面发现更新失败&#xff0c;通过查看应用程序日志&#xff0c;发现发生了死锁 sql 1 如下 1.最初版本根据id为条件&#xff0c;更新&#xff08;plan_start_time 二级索引&#xff09; update tt_task …

深入MySQL死锁场景

总结死锁需满足以下条件&#xff1a; 2个或者2个以上的并发事务操作并发事务之间存在锁冲突锁冲突关系成环形 GAP锁和Insert的隐式锁&#xff0c;最容易导致死锁&#xff0c;以下分析从这俩典型场景开始。 1. 表结构 建立以下表作为场景验证&#xff0c;id为主键&#xff0…

mysql死锁查询语句

mysql 死锁:如何解决mysql死锁 可直接在mysql命令行执行&#xff1a;showengineinnodbstatus\G;查看造成死锁的sql语句&#xff0c;分析索引情况&#xff0c;然后优化sql然后showprocesslist;另外可以打开慢查询日志&#xff0c;linux下打开需在my.cnf的[mysqld]里面加上以下内…

MySQL死锁排查步骤

系列文章目录 第一章&#xff1a;sql_mode模式 第二章&#xff1a;optimize table、analyze table、alter table、gh-ost 第三章&#xff1a;InnoDB MVCC原理 第四章&#xff1a;sql语句执行过程 第五章&#xff1a;Percona Toolkit工具简介 第六章&#xff1a;MySQL索引 第七…

MySql死锁过程

死锁一般怎么导致呢&#xff0c; 抛开一堆概念&#xff0c; 我就把死锁当成死结。 就是你代码获取锁的顺序问题。 MySql的死锁和我们正常代码也一样&#xff0c; 都是互通的&#xff0c; 当你修改一个表的行数据的时候&#xff0c; 就需要对那一行数据进行加锁。 所以很容易想…

中秋遇到mysql死锁怎么办

文章目录 前言一、什么是死锁二、死锁的产生条件三、死锁示例四、死锁的分析和查看1.查看最近1个死锁信息2.查看正在运行中的事务信息3.查看加锁信息 五、死锁的内部处理方案1.死锁探测机制2.锁等待超时机制 六、手动释放锁1.表级锁手动释放2.行级锁手动释放 七、死锁的优化策略…

mysql 死锁分析

一、 什么是死锁 死锁是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去.此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等的进程称为死锁进程.二、 死锁产生的四个必要条件 1.互斥条件&#xff1a;指进…

MySQL死锁分析

背景知识 MySQL数据库InnoDB引擎的行级锁在使用时是在索引记录上加锁的。 行级锁从占有模式上分为&#xff1a; 排他锁&#xff1a;独占行数据&#xff0c;如某事务获取了该行记录的排他锁&#xff0c;其他事务在获取该记录的排他锁和共享锁时需等待&#xff1b;共享锁&…

故障分析 | MySQL死锁案例分析

作者&#xff1a;杨奇龙 网名“北在南方”&#xff0c;资深 DBA&#xff0c;主要负责数据库架构设计和运维平台开发工作&#xff0c;擅长数据库性能调优、故障诊断。 本文来源&#xff1a;原创投稿 *爱可生开源社区出品&#xff0c;原创内容未经授权不得随意使用&#xff0c;转…

MySQL死锁

参考博客&#xff1a; https://blog.csdn.net/sinat_41653656/article/details/109629094 Mysql 锁类型和加锁分析 MySQL有三种锁的级别&#xff1a;页级、表级、行级。 表级锁&#xff1a;开销小&#xff0c;加锁快&#xff1b;不会出现死锁&#xff1b;锁定粒度大&#xff…

mysql死锁语句_Mysql死锁

笔者最近在生产环境错误日志上看到updating database. Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLTransactionRollbackException: Deadlock found when trying to get lock; try restarting transaction 这样的日志 ,网上看了很多文章 发现这篇文章 跟自己的场景非常接近…