一、概述
队列,又称为伫列(queue),是先进先出(FIFO, First-In-First-Out)的线性表。在具体应用中通常用链表或者数组来实现。队列只允许在后端(称为rear)进行插入操作,在前端(称为front)进行删除操作。队列的操作方式和堆栈类似,唯一的区别在于队列只允许新数据在后端进行添加。
在Java中队列又可以分为两个大类,一种是阻塞队列和非阻塞队列。
1、没有实现阻塞接口:
1)实现java.util.Queue的LinkList,
2)实现java.util.AbstractQueue接口内置的不阻塞队列: PriorityQueue 和 ConcurrentLinkedQueue
2、实现阻塞接口的
java.util.concurrent 中加入了 BlockingQueue 接口和五个阻塞队列类。它实质上就是一种带有一点扭曲的 FIFO 数据结构。不是立即从队列中添加或者删除元素,线程执行操作阻塞,直到有空间或者元素可用。
五个队列所提供的各有不同:
* ArrayBlockingQueue :一个由数组支持的有界队列。
* LinkedBlockingQueue :一个由链接节点支持的可选有界队列。
* PriorityBlockingQueue :一个由优先级堆支持的*优先级队列。
* DelayQueue :一个由优先级堆支持的、基于时间的调度队列。
* SynchronousQueue :一个利用 BlockingQueue 接口的简单聚集(rendezvous)机制。
队列是Java中常用的数据结构,比如在线程池中就是用到了队列,比如消息队列等。
由队列先入先出的特性,我们知道队列数据的存储结构可以两种,一种是基于数组实现的,另一种则是基于单链实现。前者在创建的时候就已经确定了数组的长度,所以队列的长度是固定的,但是可以循环使用数组,所以这种队列也可以称之为循环队列。后者实现的队列内部通过指针指向形成一个队列,这种队列是单向且长度不固定,所以也称之为非循环队列。下面我将使用两种方式分别实现队列。
二、基于数组实现循环队列
由于在往队列中放数据或拉取数据的时候需要移动数组对应的下标,所以需要记录一下队尾和队头的位置。说一下几个核心的属性吧:
1、queue:队列,object类型的数组,用于存储数据,长度固定,当存储的数据数量大于数组当度则抛出异常;
2、head:队头指针,int类型,用于记录队列头部的位置信息。
3、tail:队尾指针,int类型,用于记录队列尾部的位置信息。
4、size:队列长度,队列长度大于等于0或者小于等于数组长度。
/**
* 队列管道,当管道中存放的数据大于队列的长度时将不会再offer数据,直至从队列中poll数据后
*/
private Object[] queue;
//队列的头部,获取数据时总是从头部获取
private int head;
//队列尾部,push数据时总是从尾部添加
private int tail;
//队列长度
private int size;
//数组中能存放数据的最大容量
private final static int MAX_CAPACITY = <<;
//数组长度
private int capacity;
//最大下标
private int maxIndex;
三、数据结构
图中,红色部分即为队列的长度,数组的长度为16。因为这个队列是循环队列,所以队列的头部不一定要在队列尾部前面,只要队列的长度不大于数组的长度就可以了。
四、构造方法
1、MyQueue(int initialCapacity):创建一个最大长度为 initialCapacity的队列。
2、MyQueue():创建一个默认最大长度的队列,默认长度为16;
public MyQueue(int initialCapacity){
if (initialCapacity > MAX_CAPACITY)
throw new OutOfMemoryError("initialCapacity too large");
if (initialCapacity <= )
throw new IndexOutOfBoundsException("initialCapacity must be more than zero");
queue = new Object[initialCapacity];
capacity = initialCapacity;
maxIndex = initialCapacity - ;
head = tail = -;
size = ;
}
public MyQueue(){
queue = new Object[];
capacity = ;
head = tail = -;
size = ;
maxIndex = ;
}
五、往队列添加数据
添加数据时,首先判断队列的长度是否超出了数组的长度,如果超出了就添加失败(也可以设置成等待,等到有人消费了队列里的数据,然后再添加进去)。都是从队列的尾部添加数据的,添加完数据后tail指针也会相应自增1。具体实现如一下代码:
/**
* 往队列尾部添加数据
* @param object 数据
*/
public void offer(Object object){
if (size >= capacity){
System.out.println("queue's size more than or equal to array's capacity");
return;
}
if (++tail > maxIndex){
tail = ;
}
queue[tail] = object;
size++;
}
六、向队列中拉取数据
拉取数据是从队列头部拉取的,拉取完之后将该元素删除,队列的长度减一,head自增1。代码如下:
/**
* 从队列头部拉出数据
* @return 返回队列的第一个数据
*/
public Object poll(){
if (size <= ){
System.out.println("the queue is null");
return null;
}
if (++head > maxIndex){
head = ;
}
size--;
Object old = queue[head];
queue[head] = null;
return old;
}
七、其他方法
1、查看数据:这个方法跟拉取数据一样都是获取到队列头部的数据,区别是该方法不会将对头数据删除:
/**
* 查看第一个数据
* @return
*/
public Object peek(){
return queue[head];
}
2、清空队列:
/**
* 清空队列
*/
public void clear(){
for (int i = ; i < queue.length; i++) {
queue[i] = null;
}
tail = head = -;
size = ;
}
完整的代码如下:
/**
* 队列
*/
public class MyQueue { /**
* 队列管道,当管道中存放的数据大于队列的长度时将不会再offer数据,直至从队列中poll数据后
*/
private Object[] queue;
//队列的头部,获取数据时总是从头部获取
private int head;
//队列尾部,push数据时总是从尾部添加
private int tail;
//队列长度
private int size;
//数组中能存放数据的最大容量
private final static int MAX_CAPACITY = <<;
//数组长度
private int capacity;
//最大下标
private int maxIndex; public MyQueue(int initialCapacity){
if (initialCapacity > MAX_CAPACITY)
throw new OutOfMemoryError("initialCapacity too large");
if (initialCapacity <= )
throw new IndexOutOfBoundsException("initialCapacity must be more than zero");
queue = new Object[initialCapacity];
capacity = initialCapacity;
maxIndex = initialCapacity - ;
head = tail = -;
size = ;
}
public MyQueue(){
queue = new Object[];
capacity = ;
head = tail = -;
size = ;
maxIndex = ;
} /**
* 往队列尾部添加数据
* @param object 数据
*/
public void offer(Object object){
if (size >= capacity){
System.out.println("queue's size more than or equal to array's capacity");
return;
}
if (++tail > maxIndex){
tail = ;
}
queue[tail] = object;
size++;
} /**
* 从队列头部拉出数据
* @return 返回队列的第一个数据
*/
public Object poll(){
if (size <= ){
System.out.println("the queue is null");
return null;
}
if (++head > maxIndex){
head = ;
}
size--;
Object old = queue[head];
queue[head] = null;
return old;
} /**
* 查看第一个数据
* @return
*/
public Object peek(){
return queue[head];
} /**
* 队列中存储的数据量
* @return
*/
public int size(){
return size;
} /**
* 队列是否为空
* @return
*/
public boolean isEmpty(){
return size == ;
} /**
* 清空队列
*/
public void clear(){
for (int i = ; i < queue.length; i++) {
queue[i] = null;
}
tail = head = -;
size = ;
} @Override
public String toString() {
if (size <= ) return "{}";
StringBuilder builder = new StringBuilder(size + );
builder.append("{");
int h = head;
int count = ;
while (count < size){
if (++h > maxIndex) h = ;
builder.append(queue[h]);
builder.append(", ");
count++;
}
return builder.substring(,builder.length()-) + "}";
}
}
八、总结:
1、该队列为非线程安全的,在多线程环境中可能会发生数据丢失等问题。
2、队列通过移动指针来确定数组下标的位置,因为是基于数组实现的,所以队列的长度不能够超过数组的长度。
3、该队列是循环队列,这就意味着数组可以重复被使用,避免了重复创建对象带来的性能的开销。
4、添加数据时总是从队列尾部添加,拉取数据时总是从队列头部拉取,拉取完将对象元素删除。
欢迎大家关注公众号: 【java解忧杂货铺】,里面会不定时发布一些技术博客;关注即可免费领取大量最新,最流行的技术教学视频:
教你如何使用Java手写一个基于数组实现的队列的更多相关文章
-
教你如何使用Java手写一个基于链表的队列
在上一篇博客[教你如何使用Java手写一个基于数组的队列]中已经介绍了队列,以及Java语言中对队列的实现,对队列不是很了解的可以我上一篇文章.那么,现在就直接进入主题吧. 这篇博客主要讲解的是如何使 ...
-
java手写的动态数组JimisunArray
/** * @Author:jimisun * @Description: * @Date:Created in 22:10 2018-07-18 * @Modified By: */ public ...
-
手把手教你手写一个最简单的 Spring Boot Starter
欢迎关注微信公众号:「Java之言」技术文章持续更新,请持续关注...... 第一时间学习最新技术文章 领取最新技术学习资料视频 最新互联网资讯和面试经验 何为 Starter ? 想必大家都使用过 ...
-
【spring】-- 手写一个最简单的IOC框架
1.什么是springIOC IOC就是把每一个bean(实体类)与bean(实体了)之间的关系交给第三方容器进行管理. 如果我们手写一个最最简单的IOC,最终效果是怎样呢? xml配置: <b ...
-
搞定redis面试--Redis的过期策略?手写一个LRU?
1 面试题 Redis的过期策略都有哪些?内存淘汰机制都有哪些?手写一下LRU代码实现? 2 考点分析 1)我往redis里写的数据怎么没了? 我们生产环境的redis怎么经常会丢掉一些数据?写进去了 ...
-
利用SpringBoot+Logback手写一个简单的链路追踪
目录 一.实现原理 二.代码实战 三.测试 最近线上排查问题时候,发现请求太多导致日志错综复杂,没办法把用户在一次或多次请求的日志关联在一起,所以就利用SpringBoot+Logback手写了一个简 ...
-
java 手写 jvm高性能缓存
java 手写 jvm高性能缓存,键值对存储,队列存储,存储超时设置 缓存接口 package com.ws.commons.cache; import java.util.function.Func ...
-
看年薪50W的架构师如何手写一个SpringMVC框架
前言 做 Java Web 开发的你,一定听说过SpringMVC的大名,作为现在运用最广泛的Java框架,它到目前为止依然保持着强大的活力和广泛的用户群. 本文介绍如何用eclipse一步一步搭建S ...
-
摊牌了!我要手写一个“Spring Boot”
目前的话,已经把 Spring MVC 相关常用的注解比如@GetMapping .@PostMapping .@PathVariable 写完了.我也已经将项目开源出来了,地址:https://gi ...
随机推荐
-
net-force.nl/programming writeup
从 wechall.net 到 net-force.nl 网站,发现网站的内容不错,里面也有不同类型的挑战题目:Javascript / Java Applets / Cryptography / E ...
-
Entity Framework 通过Lambda表达式更新指定的字段
本来需要EF来更新指定的字段,后来在园子里找到了代码 var StateEntry = ((IObjectContextAdapter)dbContext).ObjectContext.ObjectS ...
-
strong和copy的区别
问题描述 在定义一个类的property时候,为property选择strong还是copy特别注意和研究明白的,如果property是NSString或者NSArray及其子类的时候,最好选择使用c ...
-
学习记录 java泛型资料
java泛型资料: 1. 概述在引入范型之前,Java类型分为原始类型.复杂类型,其中复杂类型分为数组和类.引入范型后,一个复杂类型就可以在细分成更多的类型.例如原先的类型List,现在在细分成Lis ...
-
Antelope与 Barracude MYSQL 文件格式
作者:吴炳锡 来源:http://www.mysqlsupport.cn/ 联系方式: wubingxi#163.com 转载请注明作/译者和出处,并且不能用于商业用途,违者必究. Antelope是 ...
-
Codeforces_776E: The Holmes Children (数论 欧拉函数)
题目链接 先看题目中给的函数f(n)和g(n) 对于f(n),若自然数对(x,y)满足 x+y=n,且gcd(x,y)=1,则这样的数对对数为f(n) 证明f(n)=phi(n) 设有命题 对任意自然 ...
-
BZOJ 1486: [HNOI2009]最小圈 [01分数规划]
裸题...平均权值最小的环.... 注意$dfs-spfa$时$dfs(cl)$...不要写成$dfs(u)$ #include <iostream> #include <cstdi ...
-
Day 11 作业题
1.整理装饰器的形成过程,背诵装饰器的固定格式 固定格式 def wrapper(func): def inner(*args, **kwargs): #执行函数前进行的操作 ret = func(* ...
-
常见的mysql数据库sql语句的编写和运行结果
省份城市试题#省份表 -> select * from province;+----+----------+| id | province |+----+----------+| 1 | ...
-
【C#】日期格式转换
C#里内置的DateTime基本上都可以实现这些功能,巧用DateTime会使你处理这些事来变轻松多了今天DateTime.Now.Date.ToShortDateString();昨天,就是今天的日 ...