【化解数据结构】详解队列,优先队列,循环队列,并实现一个队列

简介: 【化解数据结构】详解队列,优先队列,循环队列,并实现一个队列

image.png

大家好,我是小丞同学,一名大二的前端爱好者


这篇文章将讲解数据结构中的队列


非常感谢你的阅读,不对的地方欢迎指正


愿你忠于自己,热爱生活


知识点抢先看

什么是队列?

队列有哪些方法?

手写实现一个队列

优先队列,循环队列

LeetCode 实战

碎碎念


在上一篇文章中,我们讲了栈数据结构,它是一个线性结构,具有后进先出的特点。


在这一篇文章中,我们将讲队列数据结构,同样的它也是一个线性结构,但是它和栈有很大的不同


一、什么是队列?

和栈非常的相似,但是队列遵循的规则和栈不同


队列遵循先进先出的规则,也就是在尾部添加元素,从头部移除元素,最新添加的元素排在末尾


我们可以很形象的讲队列结构描绘成一个队伍


如下图,有很多人来买薯条,新来的人永远排在队伍的最后一位,买好的从队伍的最前面走掉

image.png

在生活中,几乎所有和排队有关的例子都可以用来描述一个队列

在上述的例子中,

我们把队伍的第一个元素称为对头,新增元素的操作叫做入队,买完薯条移除元素的操作叫做出队

在前端世界中,也有着很多关于队列的应用,例如

和事件处理机制有关的任务队列

JavaScript 在执行是会维护一个微任务队列,遇到微任务会将其加入任务队列当中,执行完宏任务后,会到任务队列中取微任务来执行。具体关于执行机制、事件循环的内容,可以看之前的文章:JavaScript 运行机制解析

二、队列有哪些方法?

和栈结构一样,队列也有着丰富的方法,比如入队、出队、查询等…

在这里我们主要介绍以下这些

方法 含义

enqueue(element) 在队列尾部添加一个新的元素

dequeue() 移除队列的第一项,并返回

front() 返回队列中第一个元素

isEmpty() 如果队列不包含任何元素,返回 true 否则为 false

size() 返回队列中的元素个数

clear() 清空队列

print() 打印所有元素

三、手写实现一个队列

了解了队列有哪些方法,可以来实现一个简单的队列结构

和栈这种线性结构一样,我们可以使用数组来实现一个队列

数组的一个元素看作是队头

数组的最后一位看作是队尾

1. 创建一个 Queue 类

首先创建一个 queue 类,用 queueData 变量来保存数据

class Queue {
    constructor() {
        this.queueData = []
    }
}

2. 实现 enqueue 方法

enqueue 方法是在数组中新增元素,根据队列的规则应该加在队尾,因此我们可以利用数组的 push 方法来实现

enqueue(element) {
    this.queueData.push(element)
}

3. 实现 dequeue 方法

dequeue 方法是移除数组的第一位元素,也就是移除对头,可以利用数组的 shift 方法来实现,取出数组的第一个元素,并返回

dequeue() {
    return this.queueData.shift()
}

我们来看看如何使用 enqueue 和 dequeue 方法

const queue = new Queue()
queue.enqueue(2) // 入
queue.enqueue(1) // 入
queue.enqueue(5) // 入
queue.dequeue()  // 出
queue.dequeue()  // c

实现动图1.gif

4. 实现 front 方法

front 方法是返回数组的一位元素,也就是返回对头的值,可以直接利用 [0] 来获取

front() {
    return this.queueData[0]
}

5. 实现 isEmpty 方法

isEmpty 方法是用来判断队列是否为空,为空的话返回 true ,不为空返回 false

这里可以采用数组的 length 来判断是否为空

isEmpty() {
    return !this.queueData.length
}

6. 实现 size 方法

size 方法可以返回队列的长度,可以用数组的 length 方法来代替

size() {
    return this.queueData.length
}

7. 实现 clear 方法

clear 方法可以清空整个队列,可以采用重置数组的方法来实现

clear() {
    this.queueData = []
}

8. 实现 print 方法

print 方法打印队列中的所有元素,我们可以采用 toString() 方法来实现

print() {
    return this.queueData.toString()
}

9. 完整的 Queue 类

class Queue {
    constructor() {
        this.queueData = []
    }
    enqueue(element) {
        this.queueData.push(element)
    }
    dequeue() {
        return this.queueData.shift()
    }
    front() {
        return this.queueData[0]
    }
    isEmpty() {
        return !this.queueData.length
    }
    size() {
        return this.queueData.length
    }
    clear() {
        this.queueData = []
    }
    print() {
      return this.queueData.toString()
  }
}


到这里,我们已经实现了一个完整的队列结构,很轻易就能实现

在队列结构中,常常被提起的还有一个优先队列,我们再来简单的介绍一下

四、优先队列

1. 什么是优先队列?

优先队列是一种元素有优先级的队列,它的元素添加和移除都是基于优先级的,优先级高的先入队,优先级低的后入队。

在现实生活中大多数情况下都是优先队列,例如:

在医院的急诊室,医生会优先处理病情严重的患者,再处理相较弱的患者

在我们学习的时候,也应当为事情添加优先级噢

2. 实现 enqueue 方法

对于一个优先队列,它和普通队列最大的区别就在于它添加元素的方法

首先每一个元素都会有一个优先级

根据优先级值的大小来插入元素

对于一个最小优先队列而言,它是根据优先级值从小到大排列的

tips: 优先级值越小,优先级越高噢

因此我们需要改造一下 enqueue 添加元素的代码

首先我们需要创建一个优先节点类

因为,队列中的元素有值和优先级两个属性,因此用类来实现

class QueueElement {
    constructor(element, priority) {
        this.element = element
        this.priority = priority
    }
}

在创建元素的时候,我们只需要 new 一下就能创建一个有值和优先级的节点

接下来实现一个 enqueue 方法

当队列空时,直接推入队列中

不空时,我们遍历这个队列,比较它的优先级。优先级值比它高的地方插入

采用 splice 方法插入,(splice:在某个位置删除多少个元素,插入什么元素)

当插入的元素的优先级值最大时,直接推入

enqueue(element, priority) {
    let queueElement = new QueueElement(element, priority)
    // 如果队列为空直接push
    if (this.isEmpty()) {
        this.data.push(queueElement)
    } else {
        // flag 记录是否成功插入
        let flag = false
        for (let i = 0; i < this.data.length; i++) {
            if (queueElement.priority < data[i].priority) {
                // 在指定位置插入
                this.data.splice(i, 0, queueElement)
                // 标记重置
                flag = true
                // 提前结束循环
                break;
            }
        }
        if(!flag) this.data.push(queueElement)
    }
}

这样一个优先队列就实现了,其他方法和普通队列一致


五、循环队列

另一个修改版的队列:循环队列。循环队列就是一圈一圈的,首尾相连的


它和普通队列的区别就是循环队列头尾相连


我们通过一个经典的击鼓传花游戏来介绍


游戏规则:


在这个游戏中,孩子们围成一个圆圈,把花尽快地传递给旁边的人。某一时刻传花停止,

这个时候花在谁手里,谁就退出圆圈结束游戏。重复这个过程,直到只剩一个孩子(胜者)。

image.png

类似于上图,输入的数字是 7 ,第一轮 c 淘汰,花传给它的下一位,从这里重新开始计数

代码实现

function hotPotato(nameList, num) {
    const queue = new Queue()
    // 添加游戏玩家
    for (let i = 0; i < nameList.length; i++) {
        queue.enqueue(nameList[i])
    }
    let dead = ''
    // 实现循环
    while (queue.size() > 1) {
        // 队列重排
        for (let i = 0; i < num; i++) {
            queue.enqueue(queue.dequeue())
        }
        // 输出淘汰者信息
        dead = queue.dequeue()
        console.log(dead + '淘汰');
    }
    // 最终返回最后一个胜利者
    return queue.dequeue()
}
let names = ['a', 'b', 'c', 'e', 'f']
let winner = hotPotato(names, 7)

这样一个击鼓传花的游戏就设计好了,你知道最终的赢家是谁吗?

六、LeetCode 实战

933. 最近的请求次数

写一个 RecentCounter 类来计算特定时间范围内最近的请求。

请你实现 RecentCounter 类:

RecentCounter() 初始化计数器,请求数为 0 。

int ping(int t) 在时间 t 添加一个新请求,其中 t 表示以毫秒为单位的某个时间,并返回过去 3000 毫秒内发生的所有请求数(包括新请求)。确切地说,返回在 [t-3000, t] 内发生的请求数。

保证 每次对 ping 的调用都使用比之前更大的 t 值。

解题思路

将每次输入的时间 t 加入到队列当中

从队列的首位元素开始,踢出不在 3000 范围内的元素

因为 t 表示的是时刻

AC 代码

var RecentCounter = function () {
    this.q = []
};
RecentCounter.prototype.ping = function (t) {
    this.q.push(t)
    // 判断对头,踢出所有不在 3000 内的元素
    while (this.q[0] < t - 3000) {
        this.q.shift()
    }
    return this.q.length
};

总结

在这篇文章中,我们从实现一个普通队列开始,将来优先队列,循环队列,最后 AC 了一道算法题,还是很有收益的~大概需要掌握以下内容


实现一个普通队列

了解如何封装优先队列的添加方法

掌握循环队列的奥秘

相关文章
|
2月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
287 9
|
13天前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
128 75
|
13天前
|
存储 C++ 索引
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
【数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】初始化队列、销毁队列、判断队列是否为空、进队列、出队列等。本关任务:编写一个程序实现环形队列的基本运算。(6)出队列序列:yzopq2*(5)依次进队列元素:opq2*(6)出队列序列:bcdef。(2)依次进队列元素:abc。(5)依次进队列元素:def。(2)依次进队列元素:xyz。开始你的任务吧,祝你成功!(4)出队一个元素a。(4)出队一个元素x。
35 13
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
|
13天前
|
存储 C语言 C++
【C++数据结构——栈与队列】链栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现链栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储整数,最大
35 9
|
13天前
|
C++
【C++数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】
【数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】(1)遇到左括号:进栈Push()(2)遇到右括号:若栈顶元素为左括号,则出栈Pop();否则返回false。(3)当遍历表达式结束,且栈为空时,则返回true,否则返回false。本关任务:编写一个程序利用栈判断左、右圆括号是否配对。为了完成本关任务,你需要掌握:栈对括号的处理。(1)遇到左括号:进栈Push()开始你的任务吧,祝你成功!测试输入:(()))
29 7
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
89 5
|
2月前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
3月前
【数据结构】-- 栈和队列
【数据结构】-- 栈和队列
23 0
|
3月前
探索数据结构:队列的的实现与应用
探索数据结构:队列的的实现与应用
|
3月前
|
存储 C语言
栈和队列题目练习
栈和队列题目练习
30 0

热门文章

最新文章