数据结构之队列
概述
在java5中新增加了java.util.Queue接口,用以支持队列的常见操作。Queue接口与List、Set同一级别,都是继承了Collection接口。 Queue使用时要尽量避免Collection的add()和remove()方法,而是要使用offer()来加入元素,使用poll()来获取并移出元素。它们的优 点是通过返回值可以判断成功与否,add()和remove()方法在失败的时候会抛出异常。 如果要使用前端而不移出该元素,使用 element()或者peek()方法。
代码实现
值得注意的是LinkedList类实现了Queue接口,因此我们可以把LinkedList当成Queue来用。
我们知道,在数组中,若知道数据项的下标,便可立即访问该数据项,或者通过顺序搜索数据项,访问到数组中的各个数据项。但是栈和队列不同,它们的访问是受限制的,即在特定时刻只有一个数据项可以被读取或者被删除。众所周知,栈是先进后出,只能访问栈顶的数据,队列是先进先出,只能访问头部数据。这里不再赘述。
栈的主要机制可以用数组来实现,也可以用链表来实现,下面用数组来实现栈的基本操作:
public class ArrayStack {
private long[] a;
private int size; //栈数组的大小
private int top; //栈顶
public ArrayStack(int maxSize) {
this.size = maxSize;
this.a = new long[size];
this.top = -1; //表示空栈
}
public void push(long value) {//入栈
if(isFull()) {
System.out.println("栈已满!");
return;
}
a[++top] = value;
}
public long peek() {//返回栈顶内容,但不删除
if(isEmpty()) {
System.out.println("栈中没有数据");
return 0;
}
return a[top];
}
public long pop() { //弹出栈顶内容,删除
if(isEmpty()) {
System.out.println("栈中没有数据!");
return 0;
}
return a[top--];
}
public int size() {
return top + 1;
}
public boolean isEmpty() {
return (top == -1);
}
public boolean isFull() {
return (top == size -1);
}
public void display() {
for(int i = top; i >= 0; i--) {
System.out.print(a[i] + " ");
}
System.out.println("");
}
}
数据项入栈和出栈的时间复杂度均为O(1)。这也就是说,栈操作所消耗的时间不依赖于栈中数据项的个数,因此操作时间很短。栈不需要比较和移动操作。
队列也可以用数组来实现,不过这里有个问题,当数组下标满了后就不能再添加了,但是数组前面由于已经删除队列头的数据了,导致空。所以队列我们可以用循环数组来实现,见下面的代码:
public class RoundQueue {
private long[] a;
private int size; //数组大小
private int nItems; //实际存储数量
private int front; //头
private int rear; //尾
public RoundQueue(int maxSize) {
this.size = maxSize;
a = new long[size];
front = 0;
rear = -1;
nItems = 0;
}
public void insert(long value) {
if(isFull()){
System.out.println("队列已满");
return;
}
rear = ++rear % size;
a[rear] = value; //尾指针满了就循环到0处,这句相当于下面注释内容
nItems++;
/* if(rear == size-1){
rear = -1;
}
a[++rear] = value;
*/
}
public long remove() {
if(isEmpty()) {
System.out.println("队列为空!");
return 0;
}
nItems--;
front = front % size;
return a[front++];
}
public void display() {
if(isEmpty()) {
System.out.println("队列为空!");
return;
}
int item = front;
for(int i = 0; i < nItems; i++) {
System.out.print(a[item++ % size] + " ");
}
System.out.println("");
}
public long peek() {
if(isEmpty()) {
System.out.println("队列为空!");
return 0;
}
return a[front];
}
public boolean isFull() {
return (nItems == size);
}
public boolean isEmpty() {
return (nItems == 0);
}
public int size() {
return nItems;
}
}
和栈一样,队列中插入数据项和删除数据项的时间复杂度均为O(1)。
还有个优先级队列,优先级队列是比栈和队列更专用的数据结构。优先级队列与上面普通的队列相比,主要区别在于队列中的元素是有序的,关键字最小(或者最大)的数据项总在队头。数据项插入的时候会按照顺序插入到合适的位置以确保队列的顺序。优先级队列的内部实现可以用数组或者一种特别的树——堆来实现。堆可参考第8节内容。这里用数组实现优先级队列。
public class PriorityQueue {
private long[] a;
private int size;
private int nItems;//元素个数
public PriorityQueue(int maxSize) {
size = maxSize;
nItems = 0;
a = new long[size];
}
public void insert(long value) {
if(isFull()){
System.out.println("队列已满!");
return;
}
int j;
if(nItems == 0) { //空队列直接添加
a[nItems++] = value;
}
else{//将数组中的数字依照下标按照从大到小排列
for(j = nItems-1; j >= 0; j--) {
if(value > a[j]){
a[j+1] = a[j];
}
else {
break;
}
}
a[j+1] = value;
nItems++;
}
}
public long remove() {
if(isEmpty()){
System.out.println("队列为空!");
return 0;
}
return a[--nItems];
}
public long peekMin() {
return a[nItems-1];
}
public boolean isFull() {
return (nItems == size);
}
public boolean isEmpty() {
return (nItems == 0);
}
public int size() {
return nItems;
}
public void display() {
for(int i = nItems-1; i >= 0; i--) {
System.out.print(a[i] + " ");
}
System.out.println(" ");
}
}
- org.thymeleaf.exceptions.TemplateProcessingException: Exception evaluating SpringEL expression
- 巧用分析函数循序渐进解决实际问题 (r4笔记第10天)
- 支持多用户web终端实现及安全保障(nodejs)
- 你看到的最直白清晰的CNN讲解
- oracle中的数组(第一篇)(r4笔记第9天)
- org.springframework.expression.spel.SpelEvaluationException: EL1004E: Method call: Method service()
- GITCHAT系列2:个性化推荐
- org.springframework.expression.spel.SpelEvaluationException: EL1011E: Method call: Attempted to call
- 【Keras】完整实现‘交通标志’分类、‘票据’分类两个项目,让你掌握深度学习图像分类
- PaddlePaddle发布新版API,简化深度学习编程
- 13(01)总结StringBuffer,StringBuilder,数组高级,Arrays,Integer,Character
- thymeleaf模板引擎调用java类中的方法(附源码)
- 由hugepage设置导致的数据库事故(r4笔记第28天)
- 判断js引擎是javascriptCore或者v8
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- TypeScript: 为什么必须学
- 一、环境搭建、以及聊聊更重要的...
- 四、作用域与作用域链
- 【从0到1学算法】递归
- 手把手教你创建 Spring MVC 实例
- 举一反三:三种问题,两个指针,一种方法
- torch.backends.cudnn.benchmark ?!
- jQuery ui中sortable draggable droppable的使用
- 阿里面试:看你springBoot用的比较溜来,说说springboot自动装配是怎么回事?
- CVPR:深度无监督跟踪
- Discourse 数据分发
- Discourse 如何备份到 S3
- Solr集群(即SolrCloud)搭建与使用
- 在CentOS8上安装Surelog
- sqlplus执行错误的问题探究