Smarter Python数据结构:用队列实现一个排序系统(关于人员入队列与出队列)

简介: Smarter Python数据结构:用队列实现一个排序系统(关于人员入队列与出队列)

简说Python,号主老表,Python终身学习者,数据分析爱好者,从18年开始分享Python知识,原创文章227篇,写过Python、SQL、Excel入门文章,也写过Web开发、数据分析文章,老表还总结整理了一份2022Python学习资料和电子书资源,关注后私信回复:2022 即可领取。

今天给大家分享的书籍《Python程序员面试算法宝典》第二章第七小节:用队列实现一个排序系统(关于人员入队列与出队列)。

如果你是第一次看,也许,你可以看看本系列下面的文章:

Python数据结构:链表合集(12+7),复现几遍,包你学会

Smaller And Smarter Python数据结构:自己动手实现栈

Smaller Python数据结构:自己动手实现队列

Smarter Python数据结构:如何翻转栈

Smarter Python数据结构:根据入栈序列来判断可能的出栈序列

Smarter Python数据结构:利用O(1)的时间复杂度求栈中最小元素

Smarter Python数据结构:如何用两个栈模拟队列


今日问题

"""
目标:写一个程序,设计一个排队系统
能让每个进入队伍的用户都能看到自己在队列中所处位置的变化。
输入:进入a, b
输出::id: 1 name: a  id: 2 name: b
Goal: write a program and design a queuing system
It enables each user entering the queue to see the change of their
 position in the queue.
Input: enter a, B
Output:: ID: 1 Name: a ID: 2 Name: B
"""
要求

1、进入队列的每个用户可以看到自己的信息

2、队伍可能随时有人加入和退出

3、当有人退出时,需要重新给用户进行排序

解题

首先我们之前已经写好了栈的基本操作,在b_2_implementation_queue.py文件中,所以我们先导入这个包。

from StackQueueHash.b_2_implementation_queue import LinkQueue

方法:基本法-List队列

"""
Method One : 基本法-List队列
核心思想:这个题不难,我们之前动手实现的队列即可完成对应功能,难在如何优化,
方法一是基本方法,我们用List队列,方便实现随机用户离队。
Method one: Basic Law list queue
Core idea: this problem is not difficult. The queue we implemented
 before can complete the corresponding functions. The difficulty
 lies in how to optimize. The first method is the basic method.
 We use list queue to facilitate the implementation of random user
 departure.
"""

代码

class User:# 一个用户类
    def __init__(self, id, name):
        self.id = id  # 用户id,唯一不可变
        self.name = name  # 用户名
        self.seq = 0# 用户入队序号
    # set,get方法
    def get_name(self):
        return self.name
    def set_name(self, name):
        self.name = name
    def get_id(self):
        return self.id
    def get_seq(self):
        return self.seq
    def set_seq(self, seq):
        self.seq = seq
    # 返回人员完整信息
    def to_string(self):
        return"id:" + str(self.id) + " " + "name:" + self.name + " " + "seq:" + str(self.seq)
class MyQueue:# 排序系统队列
    def __init__(self):
        self.q = ListQueue()  # 初始化一个队列
    def enter_queue(self, u):# 入队列
        u.set_seq(self.q.queue_len() + 1)  # 元素序列号
        self.q.queue_push(u)  # 添加元素
    def pop_queue(self, u):# 出队列,不一定是队首
        self.q.queue.remove(u)  # 出队列
        self.update_seq()  # 更新队列
    def update_seq(self):# 更新元素序列号
        i = 1
        for u in self.q.queue:
            u.set_seq(i)  # 重新设置队列序号
            i = i + 1
    # 打印队列信息
    def print_queue(self):
        for i in self.q.queue:  # 遍历打印人员信息
            print(i.to_string())
# 初始化队列,让人员入队列
def init_user_queue(user_dict, my_queue):
    for k, v in user_dict.items():  # 遍历存储了队列人员信息的字典
        user = User(v, k)  # 初始化人员对象
        my_queue.enter_queue(user)  # 入队列

测试代码

# 当然,也许还有别的方法,比如建一个辅助的链表
# 欢迎你说出你的想法
# 程序入口,测试函数功能
if __name__ == "__main__":
    user_dict = {"Hadoop": 1, "Spark": 2, "Hive": 3, "SQL": 4}  # 方便书写,我们把user信息写到字典里
    my_queue = MyQueue()  # 初始化一个队列
    init_user_queue(user_dict, my_queue)   # 初始化队列元素,人员入队列
    print("当前队列人员信息:")
    my_queue.print_queue()  # 打印入队列后所有人员
    import random
    queue_len = my_queue.q.queue_len()
    seq = random.randint(0, queue_len-1)
    user = my_queue.q.queue[seq]
    # 获取到一个随机人员
    print("离开人员信息:")
    print(user.to_string())   # 打印离队人员信息
    my_queue.pop_queue(user)  # 让获取到的随机人员出队列
    print("剩余人员信息:")    # 打印剩余人员信息
    my_queue.print_queue()

优化1

"""
优化1:前面我们用的是List实现的队列,但其效率较低,特别是当队列元素变多,
再要随机删除时,List 的底层是数组(array),其最大的时间空间消耗出现在存
储元素增长超过当前数组分配的大小时,所有元素都必须移动到新的位置,尤其对于
头部的插入与删除(O(n)),其实Python里有种数据结构叫:deque,双队列,
Python 中的 List 类型(使用其内部的 append:尾部进,和 pop 成员函数:尾
部出,左端受限)能胜任 stack 的角色,但其在前端的 pop(或 insert)的操作
时间都是线性级的,下面我们用deque来代替list。
参考链接:https://blog.csdn.net/lanchunhui/article/details/50958069
Optimization 1: We used the queue implemented by list before, but
 its efficiency is low. Especially when there are more queue elements
 and they need to be deleted randomly, the bottom layer of list is array.
 The largest time and space consumption occurs when the storage elements
 grow larger than the current array allocation. All elements must be moved
 to a new location, especially for the insertion and deletion of the header
 (o(n)), in fact, there is a data structure in Python called deque, double
 queue. The list type in python (using its internal append: tail in, and pop
 member function: tail out, left end Limited) is competent for the role of stack,
 but the operation time of its pop (or insert) in the front end is linear. Next,
 we use deque to replace list.
Reference link: https://blog.csdn.net/lanchunhui/article/details/50958069
"""

代码

from collections import deque
class MyQueue1:# 排序系统队列
    def __init__(self):
        self.q = deque()  # 初始化一个队列,双向队列
    def enter_queue(self, u):# 入队列
        u.set_seq(len(self.q) + 1)  # 元素序列号
        self.q.append(u)  # 添加元素
    def pop_queue(self):# 出队列
        self.q.popleft()  # 出队列,左边出(队首)
        self.update_seq()  # 更新队列
    def random_pop_queue(self, u):# 随机出队列
        self.q.remove(u)  # 出队列
        self.update_seq()  # 更新队列
    def update_seq(self):# 更新元素序列号
        i = 1
        for u in self.q:
            u.set_seq(i)  # 重新设置队列序号
            i = i + 1
    # 打印队列信息
    def print_queue(self):
        for i in self.q:  # 遍历打印人员信息
            print(i.to_string())
# 初始化队列,让人员入队列
def init_user_queue(user_dict, my_queue):
    for k, v in user_dict.items():  # 遍历存储了队列人员信息的字典
        user = User(v, k)  # 初始化人员对象
        my_queue.enter_queue(user)  # 入队列

测试代码

# 当然,也许还有别的方法,比如建一个辅助的链表
# 欢迎你说出你的想法
# 程序入口,测试函数功能
if __name__ == "__main__":
    user_dict = {"Hadoop": 1, "Spark": 2, "Hive": 3, "SQL": 4}  # 方便书写,我们把user信息写到字典里
    my_queue = MyQueue1()  # 初始化一个队列
    init_user_queue(user_dict, my_queue)   # 初始化队列元素,人员入队列
    print("当前队列人员信息:")
    my_queue.print_queue()  # 打印入队列后所有人员
    import random
    queue_len = len(my_queue.q)
    seq = random.randint(0, queue_len-1)
    user = my_queue.q[seq]
    # 获取到一个随机人员
    print("离开人员信息:")
    print(user.to_string())   # 打印离队人员信息
    my_queue.random_pop_queue(user)  # 让获取到的随机人员出队列
    print("剩余人员信息:")    # 打印剩余人员信息
    my_queue.print_queue()

运行结果

image.png

注意哈,这两个方法,其实差不多,只是一个队列是用的自己写的(性能肯定差些),一个是环境里的(双队列,用起来更加灵活),两个测试代码也是有差别的,所以我写了两个,希望大家学习过程中不要搞混了。

本文代码思路来自书籍《Python程序员面试宝典》,书中部分代码有问题,文中已经修改过了,并添加上了丰厚的注释,方便大家学习,后面我会把所有内容开源放到Github上,包括代码,思路,算法图解(手绘或者电脑画),时间充裕的话,会录制视频。希望大家多多支持。

大家好,我是老表 觉得本文不错的话,转发、留言、点赞,是对我最大的支持。

欢迎关注微信公众号:简说Python 关注后回复:1024,可以领取精选编程学习电子书籍。

相关文章
|
2月前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
167 77
|
19天前
|
算法 调度 C++
STL——栈和队列和优先队列
通过以上对栈、队列和优先队列的详细解释和示例,希望能帮助读者更好地理解和应用这些重要的数据结构。
26 11
|
1月前
|
DataX
☀☀☀☀☀☀☀有关栈和队列应用的oj题讲解☼☼☼☼☼☼☼
### 简介 本文介绍了三种数据结构的实现方法:用两个队列实现栈、用两个栈实现队列以及设计循环队列。具体思路如下: 1. **用两个队列实现栈**: - 插入元素时,选择非空队列进行插入。 - 移除栈顶元素时,将非空队列中的元素依次转移到另一个队列,直到只剩下一个元素,然后弹出该元素。 - 判空条件为两个队列均为空。 2. **用两个栈实现队列**: - 插入元素时,选择非空栈进行插入。 - 移除队首元素时,将非空栈中的元素依次转移到另一个栈,再将这些元素重新放回原栈以保持顺序。 - 判空条件为两个栈均为空。
|
3月前
|
存储 缓存 监控
局域网屏幕监控系统中的Python数据结构与算法实现
局域网屏幕监控系统用于实时捕获和监控局域网内多台设备的屏幕内容。本文介绍了一种基于Python双端队列(Deque)实现的滑动窗口数据缓存机制,以处理连续的屏幕帧数据流。通过固定长度的窗口,高效增删数据,确保低延迟显示和存储。该算法适用于数据压缩、异常检测等场景,保证系统在高负载下稳定运行。 本文转载自:https://www.vipshare.com
140 66
|
2月前
|
存储 C++ 索引
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
【数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】初始化队列、销毁队列、判断队列是否为空、进队列、出队列等。本关任务:编写一个程序实现环形队列的基本运算。(6)出队列序列:yzopq2*(5)依次进队列元素:opq2*(6)出队列序列:bcdef。(2)依次进队列元素:abc。(5)依次进队列元素:def。(2)依次进队列元素:xyz。开始你的任务吧,祝你成功!(4)出队一个元素a。(4)出队一个元素x。
67 13
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
|
4月前
|
存储 开发者 索引
Python 中常见的数据结构
这些数据结构各有特点和适用场景,在不同的编程任务中发挥着重要作用。开发者需要根据具体需求选择合适的数据结构,以提高程序的效率和性能
169 59
|
4月前
|
存储 索引 Python
Python编程数据结构的深入理解
深入理解 Python 中的数据结构是提高编程能力的重要途径。通过合理选择和使用数据结构,可以提高程序的效率和质量
182 59
|
4月前
|
存储 开发者 Python
Python 中的数据结构与其他编程语言数据结构的区别
不同编程语言都有其设计理念和应用场景,开发者需要根据具体需求和语言特点来选择合适的数据结构
133 55
|
2月前
|
存储 C语言 C++
【C++数据结构——栈与队列】链栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现链栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储整数,最大
57 9
|
2月前
|
C++
【C++数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】
【数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】(1)遇到左括号:进栈Push()(2)遇到右括号:若栈顶元素为左括号,则出栈Pop();否则返回false。(3)当遍历表达式结束,且栈为空时,则返回true,否则返回false。本关任务:编写一个程序利用栈判断左、右圆括号是否配对。为了完成本关任务,你需要掌握:栈对括号的处理。(1)遇到左括号:进栈Push()开始你的任务吧,祝你成功!测试输入:(()))
50 7

热门文章

最新文章