【恋上数据结构与算法一】(五)队列

队列(Queue)

◼ 队列是一种特殊的线性表,只能在头尾两端进行操作
队尾(rear):只能从队尾添加元素,一般叫做 enQueue,入队
队头(front):只能从队头移除元素,一般叫做 deQueue,出队
先进先出的原则,First In First Out,FIFO

队列的接口设计

◼int size(); // 元素的数量
◼boolean isEmpty(); // 是否为空
◼void clear(); // 清空
◼void enQueue(E element); // 入队
◼E deQueue(); // 出队
◼E front(); // 获取队列的头元素

◼ 队列的内部实现是否可以直接利用以前学过的数据结构?
动态数组、链表
优先使用双向链表,因为队列主要是往头尾操作元素

练习 – 用栈实现队列

◼ https://leetcode-cn.com/problems/implement-queue-using-stacks/
◼ 准备2个栈:inStack、outStack
入队时,push 到 inStack 中
出队时
✓如果 outStack 为空,将 inStack 所有元素逐一弹出,push 到 outStack,outStack 弹出栈顶元素
✓如果 outStack 不为空, outStack 弹出栈顶元素
◼ 假设如下操作:11入队、22入队、出队、33入队、出队 inStack outStack inStack outStack

package alangeit;

import alangeit.list.LinkedList;
import alangeit.list.List;

public class Queue {
    private List list = new LinkedList<>();
    
    public int size() {
        return list.size();
    }

    public boolean isEmpty() {
        return list.isEmpty();
    }
    
    public void clear() {
        list.clear();
    }

    public void enQueue(E element) {
        list.add(element);
    }

    public E deQueue() {
        return list.remove(0);
    }

    public E front() {
        return list.get(0);
    }
}
// 队列
static void test1() {
    System.out.println("--------------------------------- 队列");
    Queue queue = new Queue<>();
    queue.enQueue(11);
    queue.enQueue(22);
    queue.enQueue(33);
    queue.enQueue(44);
    
    while (!queue.isEmpty()) {
        System.out.println(queue.deQueue());
    }
}

双端队列(Deque)

◼ 双端队列是能在头尾两端添加、删除的队列
英文 deque 是 double ended queue 的简称

◼int size(); // 元素的数量
◼boolean isEmpty(); // 是否为空
◼void clear(); // 清空
◼void enQueueRear(E element);// 从队尾入队
◼E deQueueFront(); // 从队头出队
◼void enQueueFront(E element);// 从队头入队
◼E deQueueRear(); // 从队尾出队
◼E front(); // 获取队列的头元素
◼E rear(); // 获取队列的尾元素

package alangeit;

// 双端队列

import alangeit.list.LinkedList;
import alangeit.list.List;

public class Deque {
    private List list = new LinkedList<>();
    
    public int size() {
        return list.size();
    }

    public boolean isEmpty() {
        return list.isEmpty();
    }
    
    public void clear() {
        list.clear();
    }

    public void enQueueRear(E element) {
        list.add(element);
    }

    public E deQueueFront() {
        return list.remove(0);
    }

    public void enQueueFront(E element) {
        list.add(0, element);
    }

    public E deQueueRear() {
        return list.remove(list.size() - 1);
    }

    public E front() {
        return list.get(0);
    }

    public E rear() {
        return list.get(list.size() - 1);
    }
}
// 双端队列
static void test() {
    System.out.println("--------------------------------- 双端队列");
    Deque queue1 = new Deque<>();
    queue1.enQueueFront(11);
    queue1.enQueueFront(22);
    queue1.enQueueRear(33);
    queue1.enQueueRear(44);
    
    /* 头 22 11 33 44 尾 */
//        while (!queue1.isEmpty()) {
//            System.out.println(queue1.deQueueFront());
//        }
    
    /* 尾  44  33   11  22 头 */
    while (!queue1.isEmpty()) {
        System.out.println(queue1.deQueueRear());
    }
}

循环队列(Circle Queue)

◼其实队列底层也可以使用动态数组实现,并且各项接口也可以优化到 O(1) 的时间复杂度
◼ 这个用数组实现并且优化之后的队列也叫做:循环队列

◼ 循环双端队列:可以进行两端添加、删除操作的循环队列

package alangeit.circle;

@SuppressWarnings("unchecked")
public class CircleQueue {
    private int front;// 队头下标
    private int size;
    private E[] elements;
    private static final int DEFAULT_CAPACITY = 10;
    
    public CircleQueue() {
        elements = (E[]) new Object[DEFAULT_CAPACITY];
    }
    
    public int size() {
        return size;
    }

    public boolean isEmpty() {
        return size == 0;
    }
    
    public void clear() {
        for (int i = 0; i < size; i++) {
            elements[index(i)] = null;
        }
        front = 0;
        size = 0;
    }

    // 入队
    public void enQueue(E element) {
        ensureCapacity(size + 1);
        
        elements[index(size)] = element;
        size++;
    }

    // 出队
    public E deQueue() {
        E frontElement = elements[front];
        elements[front] = null;
        front = index(1);
        size--;
        return frontElement;
    }

    public E front() {
        return elements[front];
    }
    
    @Override
    public String toString() {
        StringBuilder string = new StringBuilder();
        string.append("capcacity=").append(elements.length)
        .append(" size=").append(size)
        .append(" front=").append(front)
        .append(", [");
        for (int i = 0; i < elements.length; i++) {
            if (i != 0) {
                string.append(", ");
            }
            
            string.append(elements[i]);
        }
        string.append("]");
        return string.toString();
    }
    
    private int index(int index) {
        index += front;
        return index - (index >= elements.length ? elements.length : 0);
    }
    
    /**
     * 保证要有capacity的容量
     * @param capacity
     */
    private void ensureCapacity(int capacity) {
        int oldCapacity = elements.length;
        if (oldCapacity >= capacity) return;
        
        // 新容量为旧容量的1.5倍
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        E[] newElements = (E[]) new Object[newCapacity];
        for (int i = 0; i < size; i++) {
            newElements[i] = elements[index(i)];
        }
        elements = newElements;
        
        // 重置front
        front = 0;
    }
}
// 循环队列
static void test() {
    
    System.out.println("--------------------------------- 循环队列");
    
    CircleQueue queue = new CircleQueue();
    // 0 1 2 3 4 5 6 7 8 9
    for (int i = 0; i < 10; i++) {
        queue.enQueue(i);// 入队
    }
    // null null null null null 5 6 7 8 9
    for (int i = 0; i < 5; i++) {
        queue.deQueue();// 出队
    }
    System.out.println(queue);
    while (!queue.isEmpty()) {
        System.out.println(queue.deQueue());
    }
    
    // 15 16 17 18 19 5 6 7 8 9
    for (int i = 15; i < 20; i++) {// 扩容
        queue.enQueue(i);// 入队
    }
    System.out.println(queue);
    while (!queue.isEmpty()) {
        System.out.println(queue.deQueue());
    }
    System.out.println(queue);
}

循环双端队列

package alangeit.circle;

// 循环双端队列

@SuppressWarnings("unchecked")
public class CircleDeque {
    private int front;
    private int size;
    private E[] elements;
    private static final int DEFAULT_CAPACITY = 10;
    
    public CircleDeque() {
        elements = (E[]) new Object[DEFAULT_CAPACITY];
    }
    
    public int size() {
        return size;
    }

    public boolean isEmpty() {
        return size == 0;
    }

    public void clear() {
        for (int i = 0; i < size; i++) {
            elements[index(i)] = null;
        }
        front = 0;
        size = 0;
    }

    /**
     * 从尾部入队
     * @param element
     */
    public void enQueueRear(E element) {
        ensureCapacity(size + 1);// 扩容
        
        elements[index(size)] = element;
        size++;
    }

    /**
     * 从头部出队
     * @param element
     */
    public E deQueueFront() {
        E frontElement = elements[front];
        elements[front] = null;
        front = index(1);
        size--;
        return frontElement;
    }

    /**
     * 从头部入队
     * @param element
     */
    public void enQueueFront(E element) {
        ensureCapacity(size + 1);// 扩容
        
        front = index(-1);
        elements[front] = element;
        size++;
    }

    /**
     * 从尾部出队
     * @param element
     */
    public E deQueueRear() {
        int rearIndex = index(size - 1);
        E rear = elements[rearIndex];
        elements[rearIndex] = null;
        size--;
        return rear;
    }

    // 头部
    public E front() {
        return elements[front];
    }

    // 尾部
    public E rear() {
        return elements[index(size - 1)];
    }

    @Override
    public String toString() {
        StringBuilder string = new StringBuilder();
        string.append("capcacity=").append(elements.length)
        .append(" size=").append(size)
        .append(" front=").append(front)
        .append(", [");
        for (int i = 0; i < elements.length; i++) {
            if (i != 0) {
                string.append(", ");
            }
            
            string.append(elements[i]);
        }
        string.append("]");
        return string.toString();
    }
    
    private int index(int index) {
        index += front;
        if (index < 0) {
            return index + elements.length;
        }
        return index - (index >= elements.length ? elements.length : 0);
    }
    
    /**
     * 保证要有capacity的容量
     * @param capacity
     */
    private void ensureCapacity(int capacity) {
        int oldCapacity = elements.length;
        if (oldCapacity >= capacity) return;
        
        // 新容量为旧容量的1.5倍
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        E[] newElements = (E[]) new Object[newCapacity];
        for (int i = 0; i < size; i++) {
            newElements[i] = elements[index(i)];
        }
        elements = newElements;
        
        // 重置front
        front = 0;
    }
}
// 循环双端队列
static void test() {
    
    System.out.println("--------------------------------- 循环双端队列");
    
    CircleDeque queue = new CircleDeque<>();
    // 头5 4 3 2 1  100 101 102 103 104 105 106 8 7 6 尾
    
    // 头 8 7 6  5 4 3 2 1  100 101 102 103 104 105 106 107 108 109 null null 10 9 尾
    for (int i = 0; i < 10; i++) {// 入队
        queue.enQueueFront(i + 1);
        queue.enQueueRear(i + 100);
    }
    System.out.println(queue);

    // 头 null 7 6  5 4 3 2 1  100 101 102 103 104 105 106 null null null null null null null 尾
    for (int i = 0; i < 3; i++) {// 出队
        queue.deQueueFront();
        queue.deQueueRear();
    }
    System.out.println(queue);

    // 头 11 7 6  5 4 3 2 1  100 101 102 103 104 105 106 null null null null null null 12 尾
    queue.enQueueFront(11);
    queue.enQueueFront(12);
    System.out.println(queue);
    while (!queue.isEmpty()) {
        System.out.println(queue.deQueueFront());
    }
}

%运算符优化

◼ 尽量避免使用乘*、除/、模%、浮点数运算,效率低下

◼ 已知n>=0,m>0
n%m 等价于 n – (m > n ? 0 : m) 的前提条件:n < 2m

作业 – 用队列实现栈

◼ https://leetcode-cn.com/problems/implement-stack-using-queues/

你可能感兴趣的:(【恋上数据结构与算法一】(五)队列)