- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
栈和队列是在程序设计中常见的数据类型,从数据结构的角度来讲,栈和队列也是线性表,是操作受限的线性表,它们的基本操作是线性表操作的子集,但从数据类型的角度来讲,它们与线性表又有着巨大的不同。本节将介绍队列的定义及其不同实现,并且给出队列的一些实际应用。
通过本节学习,应掌握以下内容:
队列 (Queue
) 是插入和删除操作分别被限制在序列两端的线性表(新元素从一段插入其中,则只能从另一端删除已有元素),对于队列而言,允许插入元素的一端称为队尾 (rear
),而允许取出元素的一段则称为队头 (front
)。
在队列中,数据到达的顺序很重要。最新添加的元素位于必须在队尾,在队列中时间最长的元素则排在最前面,这种排序原则被称作先进先出 (first in first out
,FIFO
),或后进后出 (last in first out
, LILO
)。
队列在现实中的例子随处可见,我们生活中就餐所需要进行的排队就是一个队列的现实例子,最早进入队列的人可以首先就餐,而后来者需要排于队尾等待。假设队列为 Q = ( a 0 , a 1 , … , a n ) Q=(a_0, a_1, …, a_n)Q=(a0,a1,…,an),则队头元素为 a 0 a_0a0,a n a_nan 为队尾元素,队列中元素按 a 0 , a 1 , … , a n a_0, a_1, …, a_na0,a1,…,an 的顺序入队 (enqueue
),出队 (dequeue
) 也只能按照这个顺序依次退出,队头元素是第一个出队 (dequeue
) 的元素,而只有 a 0 , a 1 , … , a n − 1 a_0, a_1, …, a_{n-1}a0,a1,…,an−1 在都出队后,才 a n a_nan 能退出队列。下图是一个简单的队列示例:
通过观察元素的添加和移除顺序,就可以快速理解队列所蕴含的思想。下图展示了队列元素的入队和出队过程,队列中元素的插入顺序和移除顺序是完全相同的。
除了主要的操作(入队和出队)外,队列还具有初始化、判队空和求队长度等辅助操作。具体而言,队列的抽象数据类型定义如下:
ADT Queue:
数据对象:D = a i ∣ a i ∈ D a t a T y p e , i = 1 , 2 , . . . , n , n ≥ 0 D={a_i|a_i∈DataType, i=1,2,...,n,n\geq0}D=ai∣ai∈DataType,i=1,2,...,n,n≥0
数据关系:R = < a i , a i + 1 > ∣ a i , a i + 1 ∈ D , i = 1 , 2 , . . . , n − 1 R={<a_{i},a_{i+1}>|a_i,a_{i+1}∈D,i=1,2,...,n-1}R=<ai,ai+1>∣ai,ai+1∈D,i=1,2,...,n−1
a 1 a_1a1为队头元素,a n a_nan为队尾元素
基本操作:
1. itit(): 初始化队列
创建一个空队列
2. size(): 求取并返回队列中所含元素的个数 n
若队列为空,则返回整数0
3. isempty(): 判断是否为空队列
判断队列中是否存储元素
4. enqueue(data): 入队
将元素 data 插入队尾
5. dequeue(): 出队
删除并返回队尾元素
4. head(): 取队头元素
返回队头元素,但并不删除元素
队列具有广泛的应用场景,例如:
除了以上应用外,我们在之后的学习中还将看到队列用作许多算法的辅助数据结构。
和线性表一样,队列同样有顺序存储和链式存储两种存储表示方式。
类似于顺序栈,队列的顺序存储结构利用一组地址连续的存储单元依次存放从队列头到队列尾依次存放,同时需要用两个指针 front
和 rear
分别指示队列头元素和队列尾元素的位置。初始化空队列时,front=rear=0
,当元素入队时,rear 加 1
,而元素出队时,front 加 1
,如下所示:
从以上示例中,可以很清楚地看到位于队头之前的空间被浪费了,为了解决这一问题,我们可以假设顺序队列为环状空间,将最后一个元素和第一个元素视为连续的,如下图所示:
从上图可以看出,当队列满时与队列空时,都有front=rear
,因此无法通过 front==rear
来判断队列是否已满。针对这一问题,有两种解决方法:1) 设置标志来指示队列中是否还有空闲空间;2) 放弃一个元素空间,即当 front
指针在 rear
指针的下一位时 ((rear+1)%max_size=front
) 队列为满。以下实现使用第二种方案。
同样顺序队列可以是固定长度和动态长度,当队列满时,定长顺序队列会抛出栈满异常,动态顺序队列则会动态申请空闲空间。
顺序队列的初始化需要 4 部分信息:queue
列表用于存储数据元素,max_size
用于存储 queue
列表的最大长度,以及 front
和 rear
分别用于记录队头元素和队尾元素的索引:
class Queue:
def __init__(self, max_size=5):
self.max_size = max_size
self.queue = [None] * self.max_size
self.front = 0
self.rear = 0
由于 front
和 rear
分别用于记录队头元素和队尾元素的索引,因此我们可以方便的计算出队列的长度;同时我们需要考虑队列为循环队列,front
可能大于 rear
,不能直接通过 rear-front
,我们需要利用公式计算解决此问题:
( r e a r − f r o n t + m a x _ s i z e ) % m a x _ s i z e (rear-front+max_size) % max_size(rear−front+max_size)%max_size
Python
实现如下:
def size(self):
return (self.rear-self.front+self.max_size) % self.max_size
根据 front
和 rear
的值可以方便的判断队列是否为空:
def isempty(self):
return self.rear==self.front
根据 front
和 rear
的值可以方便的判断队列是否还有空余空间:
def isfull(self):
return ((self.rear+1) % self.max_size == self.front)
入队时,需要首先判断队列中是否还有空闲空间,然后根据队列为定长顺序队列或动态顺序队列,入队操作稍有不同:
[定长顺序队列的入队操作] 如果队满,则引发异常:
def enqueue(self, data):
if not self.isfull():
self.queue[self.rear] = data
self.rear = (self.rear+1) % self.max_size
else:
raise IndexError("Full Queue Exception")
[动态顺序队列的入队操作] 如果队列满,则首先申请新空间,然后再执行入队操作:
def resize(self):
new_size = 2 * self.max_size
new_queue = [None] * new_size
for i in range(self.max_size):
new_queue[i] = self.queue[i]
self.queue = new_queue
self.max_size = new_size
def enqueue(self, data):
if self.isfull():
self.resize()
self.queue[self.rear] = data
self.rear = (self.rear+1) % self.max_size
入队的时间复杂度为 O ( 1 ) O(1)O(1)。这里需要注意的是,虽然当动态顺序队列满时,原队列中的元素需要首先复制到新队列中,然后添加新元素,但参考《顺序表及其操作实现》中顺序表追加操作的介绍,由于 n
次入队操作的总时间 T ( n ) T(n)T(n) 与 O ( n ) O(n)O(n) 成正比,因此队栈的摊销时间复杂度可以认为是 O ( 1 ) O(1)O(1)。
若队列不空,则删除并返回队头元素:
def dequeue(self):
if not self.isempty():
result = self.queue[self.front]
self.front = (self.front+1) % self.max_size
return result
else:
raise IndexError("Empty Queue Exception")
若队列不空,则返回队头元素:
def head(self):
if not self.isempty():
result = self.queue[self.front]
return result
else:
raise IndexError("Empty Queue Exception")
队列的另一种存储表示方式是使用链式存储结构,因此也常称为链队列,其中 enqueue
操作是通过在链表尾部插入元素来实现的,dequeue
操作是通过从头部删除节点来实现的。
队列的结点实现与链表并无差别:
class Node:
def __init__(self, data=None):
self.data = data
self.next = None
def __str__(self):
return str(self.data)
队列的初始化函数中,使其队头指针 front
和 rear
均指向 None
,并初始化队列长度:
class Queue:
def __init__(self):
self.front = None
self.rear = None
self.num = 0
返回 size
的值用于求取队列的长度,如果没有 size
属性,则需要遍历整个链表才能得到队列长度:
def size(self):
return self.num
根据队列的长度可以很容易的判断队列是否为空队列:
def isempty(self):
return self.num <= 0
入队时,在队尾插入新元素,并且需要将队尾指针 rear
指向新元素,如果队列为空,需要将队头指针 front
也指向此元素:
def enqueue(self, data):
node = Node(data)
if self.front is None:
self.rear = node
self.front = self.rear
else:
self.rear.next = node
self.rear = node
self.num += 1
若队列不空,则删除并返回队头元素,并且需要更新队头指针 front
指向原队头结点的后继结点,若出队元素尾队中最后一个结点,则更新队尾指针 rear
:
def dequeue(self):
if self.isempty():
raise IndexError("Empty Queue Exception")
result = self.front.data
self.front = self.front.next
self.num -= 1
if self.isempty():
self.rear = self.front
return result
若队列不空,则返回队头元素:
def head(self):
if self.isempty():
raise IndexError("Empty Queue Exception")
result = self.front.data
return result
队列的不同实现对比与栈的不同实现类似,可以参考《栈及其操作实现》。
接下来,我们首先测试上述实现的队列,以验证操作的有效性,然后利用实现的基本操作来解决实际算法问题。
首先初始化一个顺序队列 queue
,然后测试相关操作:
# 初始化一个最大长度为5的队列
q = Queue(5)
print('队列空?', q.isempty())
for i in range(4):
print('入队元素:', i)
q.enqueue(i)
print('队列满?', q.isfull())
print('队头元素:', q.head())
print('队列长度为:', q.size())
while not q.isempty():
print('出队元素:', q.dequeue())
测试程序输出结果如下:
队列空? True
入队元素: 0
入队元素: 1
入队元素: 2
入队元素: 3
# 队列中弃用一个空间,因此队列中有4个元素即满
队列满? True
队头元素: 0
队列长度为: 4
出队元素: 0
出队元素: 1
出队元素: 2
出队元素: 3
首先初始化一个链队列 queue
,然后测试相关操作:
# 初始化新队列
q = Queue()
print('队列空?', q.isempty())
for i in range(4):
print('入队元素:', i)
q.enqueue(i)
print('队头元素:', q.head())
print('队列长度为:', q.size())
while not q.isempty():
print('出队元素:', q.dequeue())
测试程序输出结果如下:
队列空? True
入队元素: 0
入队元素: 1
入队元素: 2
入队元素: 3
队头元素: 0
队列长度为: 4
出队元素: 0
出队元素: 1
出队元素: 2
出队元素: 3
[1] 考虑经典的约瑟夫斯环问题,n
个人围成一圈,从第 1
个人开始报数,第 m
个将被淘汰,重复上述过程,直到只剩下一个人,其余人都将被淘汰。
我们使用队列来模拟一个环,如下图所示,从队列的头部开始,将位于队首的人移出队列并立刻将其插入队列的尾部,之后此人会一直等待,直到其再次到达队列的头部。在出列和入列 m-1
次之后,位于队列头部的人出局(即第 m
个人),然后新一轮游戏开始;如此反复,直到队列中只剩下一个人(队列的大小为 1
)。
def Josephus(name_list, m):
queue = Queue()
for name in name_list:
queue.enqueue(name)
while queue.size() > 1:
for i in range(m-1):
queue.enqueue(queue.dequeue())
queue.dequeue()
return queue.head()
# n=6, m=5
result = Josephus(["A", "B", "C", "D", "E", "F"], 5)
print('幸存的人为', result)
程序输出结果如下:
幸存的人为 A
线性表基本概念
顺序表及其操作实现
单链表及其操作实现
栈及其操作实现
初学者 android 问题。好的,我已经成功写入文件。例如。 //获取文件名 String filename = getResources().getString(R.string.filename
我已经将相同的图像保存到/data/data/mypackage/img/中,现在我想显示这个全屏,我曾尝试使用 ACTION_VIEW 来显示 android 标准程序,但它不是从/data/dat
我正在使用Xcode 9,Swift 4。 我正在尝试使用以下代码从URL在ImageView中显示图像: func getImageFromUrl(sourceUrl: String) -> UII
我的 Ubuntu 安装 genymotion 有问题。主要是我无法调试我的数据库,因为通过 eclipse 中的 DBMS 和 shell 中的 adb 我无法查看/data/文件夹的内容。没有显示
我正在尝试用 PHP 发布一些 JSON 数据。但是出了点问题。 这是我的 html -- {% for x in sets %}
我观察到两种方法的结果不同。为什么是这样?我知道 lm 上发生了什么,但无法弄清楚 tslm 上发生了什么。 > library(forecast) > set.seed(2) > tts lm(t
我不确定为什么会这样!我有一个由 spring data elasticsearch 和 spring data jpa 使用的类,但是当我尝试运行我的应用程序时出现错误。 Error creatin
在 this vega 图表,如果我下载并转换 flare-dependencies.json使用以下 jq 到 csv命令, jq -r '(map(keys) | add | unique) as
我正在提交一个项目,我必须在其中创建一个带有表的 mysql 数据库。一切都在我这边进行,所以我只想检查如何将我所有的压缩文件发送给使用不同计算机的人。基本上,我如何为另一台计算机创建我的数据库文件,
我有一个应用程序可以将文本文件写入内部存储。我想仔细看看我的电脑。 我运行了 Toast.makeText 来显示路径,它说:/数据/数据/我的包 但是当我转到 Android Studio 的 An
我喜欢使用 Genymotion 模拟器以如此出色的速度加载 Android。它有非常好的速度,但仍然有一些不稳定的性能。 如何从 Eclipse 中的文件资源管理器访问 Genymotion 模拟器
我需要更改 Silverlight 中文本框的格式。数据通过 MVVM 绑定(bind)。 例如,有一个 int 属性,我将 1 添加到 setter 中的值并调用 OnPropertyChanged
我想向 Youtube Data API 提出请求,但我不需要访问任何用户信息。我只想浏览公共(public)视频并根据搜索词显示视频。 我可以在未经授权的情况下这样做吗? 最佳答案 YouTube
我已经设置了一个 Twilio 应用程序,我想向人们发送更新,但我不想回复单个文本。我只是想让他们在有问题时打电话。我一切正常,但我想在发送文本时显示传入文本,以确保我不会错过任何问题。我正在使用 p
我有一个带有表单的网站(目前它是纯 HTML,但我们正在切换到 JQuery)。流程是这样的: 接受用户的输入 --- 5 个整数 通过 REST 调用网络服务 在服务器端运行一些计算...并生成一个
假设我们有一个名为 configuration.js 的文件,当我们查看内部时,我们会看到: 'use strict'; var profile = { "project": "%Projec
这部分是对 Previous Question 的扩展我的: 我现在可以从我的 CI Controller 成功返回 JSON 数据,它返回: {"results":[{"id":"1","Sourc
有什么有效的方法可以删除 ios 中 CBL 的所有文档存储?我对此有疑问,或者,如果有人知道如何从本质上使该应用程序像刚刚安装一样,那也会非常有帮助。我们正在努力确保我们的注销实际上将应用程序设置为
我有一个 Rails 应用程序,它与其他 Rails 应用程序通信以进行数据插入。我使用 jQuery $.post 方法进行数据插入。对于插入,我的其他 Rails 应用程序显示 200 OK。但在
我正在为服务于发布请求的 API 调用运行单元测试。我正在传递请求正文,并且必须将响应作为帐户数据返回。但我只收到断言错误 注意:数据是从 Azure 中获取的 spec.js const accou
我是一名优秀的程序员,十分优秀!