数据结构与算法——队列

简介: 前面说完了栈,接下来再看看另一种很基础的数据结构,队列。顾名思义,队列跟我们现实生活中的排队很相似:例如我们在食堂排队打饭,先来的先打到,后来的只能一次排在后面,不允许插队。很明显,队列的操作也是受限的,插入元素(入队)只能在队尾,删除元素(出队)只能在队头。结合下面的图就很容易理解了

1. 概述


前面说完了栈,接下来再看看另一种很基础的数据结构,队列。顾名思义,队列跟我们现实生活中的排队很相似:例如我们在食堂排队打饭,先来的先打到,后来的只能一次排在后面,不允许插队。很明显,队列的操作也是受限的,插入元素(入队)只能在队尾,删除元素(出队)只能在队头。结合下面的图就很容易理解了:


9a3020f1e3be095508110bebde86bcc.png


2. 队列实现


和栈一样,队列也有两种实现方式,使用数组实现的队列叫做顺序队列,使用链表实现的队列叫做链式队列。这里我实现了链式队列,你也可以根据其思想使用数组来实现。

public class LinkedListQueue {
    private Node head;//队头节点指针
    private Node tail;//队尾节点指针
    private int size;//队列中元素个数
    public LinkedListQueue() {
        this.head = null;
        this.tail = null;
        this.size = 0;
    }
    //入队
    public boolean enqueue(String value) {
        Node node = new Node(value);
        if(tail == null) {
            head = tail = node;
            this.size ++;
            return true;
        }
        tail.next = node;
        tail = node;
        this.size ++;
        return true;
    }
    //出队列
    public String dequeue() {
        if(head == null) return null;//队列为空
        String result = head.getData();
        head = head.next;
        this.size --;
        return result;
    }
    //定义链表节点
    public static class Node{
        private String data;
        private Node next;
        public Node(String data) {
            this.data = data;
            this.next = null;
        }
        public String getData() {
            return data;
        }
    }



3. 循环队列


在使用数组实现队列的时候,会出现一个问题,就是随着队头和队尾指针的后移,就算数组中还有空间,也不能进行入队操作了。这时候我们需要进行数据搬移,性能会受到影响,而循环队列解决了这个问题。

循环队列就是将数组首尾相连,形成了一个环形:

ec16e07cf35118439ffb64e32e025cf.png

如上图,当指针 tail 到达数组末尾的时候,并不进行数据搬移,而是直接将指针向前移,到达了 0 这个位置。在进行一次入队,就变成了下面的样子:

e902a329e2c042494a1e1d04f18432d.png

可以看到,循环队列判断空的条件是 head == tail,而怎么判断队列满呢?答案是 (tail + 1)== head 的时候,队列就满了,不能看出循环队列实际上浪费了一个存储空间。下面我给出了循环队列的代码实现:

public class CircularQueue {
    private String[] items;
    private int size;
    private int head;
    private int tail;
    public CircularQueue(int capacity) {
        this.items = new String[capacity];
        this.size = 0;
        this.head = 0;
        this.tail = 0;
    }
    public CircularQueue() {
        this(16);
    }
    //入队列
    public boolean enqueue(String value) {
        if((tail + 1) % items.length == head) return false;//队列已满
        items[tail] = value;
        //head 和 tail 指针的移动不是简单的加减1了
        tail = (tail + 1) % items.length;
        this.size ++;
相关文章
|
5天前
|
算法 C语言
【数据结构与算法 经典例题】使用栈实现队列(图文详解)
【数据结构与算法 经典例题】使用栈实现队列(图文详解)
|
6天前
|
存储 算法
【数据结构和算法】--队列的特殊结构-循环队列
【数据结构和算法】--队列的特殊结构-循环队列
7 0
|
6天前
|
存储 测试技术
【数据结构】操作受限的线性表,队列的具体实现
【数据结构】操作受限的线性表,队列的具体实现
16 4
|
5天前
|
算法 C语言
【数据结构与算法 经典例题】使用队列实现栈(图文详解)
【数据结构与算法 经典例题】使用队列实现栈(图文详解)
|
6天前
|
算法
【C/数据结构和算法】:栈和队列
【C/数据结构和算法】:栈和队列
15 1
|
5天前
【海贼王的数据航海】栈和队列
【海贼王的数据航海】栈和队列
5 0
|
5天前
|
算法 程序员 数据处理
【数据结构与算法】使用单链表实现队列:原理、步骤与应用
【数据结构与算法】使用单链表实现队列:原理、步骤与应用
|
6天前
|
算法
【数据结构和算法】---栈和队列的互相实现
【数据结构和算法】---栈和队列的互相实现
7 0
|
6天前
|
算法
【数据结构和算法】--队列
【数据结构和算法】--队列
6 0
|
9天前
|
缓存 调度
栈和队列的区别
栈和队列的区别
10 0