木头127 发表于 2022-5-23 12:44

Python学习日记:队列抽象类型及其实现,幸运的基督徒、热土豆问题、回文词问题

本帖最后由 木头127 于 2022-5-23 12:48 编辑

队列及其Python实现

一、什么是队列?

1. 队列是一种有次序的数据集合,其特征是:新数据项的添加总是发生在一端(通常称为:"尾rear"端),而现存数据项的移除总发生在另一端(通常称为:“首front”端)
               当数据项加入队列,首先出现在队尾,随着队首数据项的移除,它逐渐接近队首。
               新加入的数据项必须在数据集末尾等待,而等待时间最长的数据项则是队首。
               这种次序安排的原则称为(FIFO:First-in-first-out)先进先出
               队列仅有一个入口和一个出口:不允许数据项直接插入队中,也不允许从中间移除数据项。
               队列例子:打印机面向多个用户提供服务、进程调度等等。



二、抽象数据类型Queue

抽象数据类型Queue由如下操作定义:

      Queue(): 创建一个空队列对象,返回值为Queue对象;

      enqueue(item): 将数据项item添加到队尾,无返回值;

      dequeue(): 从队首移除数据项,返回值为队首数据项,队列被修改;

      isEmpty(): 测试是否空队列,返回值为布尔值

      size(): 返回队列中数据项的个数

☆ 采用List来容纳Queue的数据项:

      将List的首端作为队列的尾端,List的末端作为队列首端,enqueue()复杂度为O(n),dequeue()复杂度为O(1). 首尾倒过来的实现,复杂度也倒过来

    class Queue:
      def __init__(self):
            self.items = []
   
      def isEmpty(self):
            return self.items == []
   
      def enqueue(self, item):
            self.items.insert(0, item)
   
      def dequeue(self):
            return self.items.pop()
   
      def size(self):
            return len(self.items)

三、队列的应用

1. 热土豆问题(约瑟夫问题、击鼓传花)
            模拟程序采用队列来存放所有参加游戏的人名,按照传递土豆的方向从队首排到队尾,游戏时,队首始终是持有土豆的人,模拟游戏开始,只需要将队首的人出队,随即再到队尾入队,算是土豆的一次传递,传递了num次后,将队首的人移除,不再入队,如此反复,直到队列中剩余一人。

    def hotPotato(namelist, num):
      """
      :param namelist: 参加游戏的人
      :param num: 每一轮传递几次
      :return:最后剩下的人
      """
      simqueue = Queue()
      for name in namelist:
            simqueue.enqueue(name)
   
      while simqueue.size() > 1:
            for i in range(num):
                simqueue.enqueue(simqueue.dequeue())
                # 一次传递,for将一次传递循环n次
            # 将队首的人挪出去不再回来
            simqueue.dequeue()
      return simqueue.dequeue()
                print(hotPotato(['Bill', 'David', 'Susan', 'Jane', 'Kent', 'Brad'], 7))

2.幸运的基督徒

    《幸运的基督徒》
    有15个基督徒和15个非基督徒在海上遇险,为了能让一部分人活下来不得不将其中15个人扔到海里面去,有个人想了个办法就是大家围成一个圈,由某个人开始从1报数,报到9的人就扔到海里面,他后面的人接着从1开始报数,报到9的人继续扔到海里面,直到扔掉15个人。由于上帝的保佑,15个基督徒都幸免于难,问这些人最开始是怎么站的,哪些位置是基督徒哪些位置是非基督徒。

    import queue
    def josephus(number, num, q):
      # number:人员总数, num:报到num的人处决q: 幸存的人数
      numbers = queue.Queue()
      for _ in range(1, number + 1):
            numbers.put(_)
            # print(numbers.get(), end='')
      print()
      while numbers.qsize() > q:
            for _ in range(num-1):
                numbers.put(numbers.get())
            numbers.get()
      for i in range(q):
            print(f'{numbers.get()}基', end=' ')
      return None
    print(josephus(30, 9, 15))
   
    # 另一种解决方法:
    def main():
      persons = * 30
      counter, index, number = 0, 0, 0
      while counter < 15:
            # 共杀掉39个人
            if persons:
                number += 1
                # 计数死亡的位置
                if number == 9:
                  # 当数到9 ,死
                  persons = False
                  counter += 1
                  number = 0
            index += 1
            # 索引向下走
            index %= 30
            # 围成一个圈
      for person in persons:
            print('基' if person else '非', end='')
   
   
    if __name__ == '__main__':
      main()
   

3.模拟算法:打印任务

多人共享一台打印机,采取“先到先服务”的队列策略来执行打印任务,在这种设定下,首要的问题就是: 在这种打印作业系统的容量有多大?在能够接收的等待时间内,系统能容纳多少用户以多高频率提交多少打印任务?

★一个具体的实例配置如下:

         一个实验室内,在任意一个小时内,大约有10名学生在场,这一个小时中,每人会发起2次左右的打印,每次1~20页。

★打印机的性能是:

      以草稿模式打印的话,每分钟10页。

      以正常模式打印的话,打印质量好,但速度下降为每分钟5页。

问题是: 怎么设定打印机的模式,让大家都不会等太久的前提下尽量提高打印质量?

      这是一个典型的决策支持问题,但无法通过规则直接计算。

      我们要用一段程序来模拟这种打印任务场景,然后对程序运行结果进行分析,以支持对打印机模式设定的决策。

如何对问题建模?

      首先对问题进行抽象,确定相关的对象和过程: 抛弃那些对问题实质没有关系的学生性别、年龄、打印机型号、打印内容、纸张大小等等众多细节。

★ 对问题抽象后的对象: 打印任务、打印队列、打印机

      打印任务的属性: 提交时间、打印页数

      打印队列的属性: 具有FIFO性质的打印任务队列

      打印机的属性: 打印速度、是否忙

★ 过程:生成和提交打印任务

      确定生成概率:实例为 每小时会有10个学生提交的20个作业,这样,概率是每180秒会有1个作业生成并提交,概率为每秒1/180

      确定打印页数:实例是1~20页,那么就是1 ~20页之间的概率相同。

\frac{20 tasks}{1 hour} * \frac{1 hour}{60 minutes} * \frac{1 minute}{60 seconds} *=\frac{1 tasks}{180 secons}

★过程:实施打印

      当前的打印作业:正在打印的作业

      打印结束倒计时:新作业开始打印时开始倒计时,回0表示打印完毕,可以处理下一个作业。

★模拟时间:

      统一的时间框架:以最小单位秒均匀流逝的时间,设定结束时间。

      同步所有过程:在一个时间单位里,对生成打印任务和实施打印两个过程各处理一次。

☆ 模拟流程:

★创建打印队列的对象

★时间按照秒的单位流逝

      按照概率生成打印作业,加入打印队列。

      若打印机空闲,且队列不空,则取出队首作业打印,记录此作业等待时间。

      若打印机忙,则按照打印速度1秒打印。

      若当前作业打印完成,则打印机进入空闲。

★时间用尽,开始统计平均等待时间。

★作业的等待时间:

      生成作业时,记录生成的时间戳。

      开始打印时,当前时间减去生成时间即可。

★作业的打印时间

      生成作业时,记录作业的页数。

      开始打印时,页数除以打印速度即可。

    import random
   
   
    class Queue:
      def __init__(self):
            self.items = []
   
      def isEmpty(self):
            return self.items == []
   
      def enqueue(self, item):
            self.items.insert(0, item)
   
      def dequeue(self):
            return self.items.pop()
   
      def size(self):
            return len(self.items)
   
   
    class Printer:
      def __init__(self, ppm):
            """
            currentTask: 打印任务
            timeRemaining: 任务倒计时
            :param ppm: 打印速度
            """
            self.pagerate = ppm
            self.currentTask = None
            self.timeRemaining = 0
   
      def tick(self):
            """ 打印 1秒!"""
            if self.currentTask is not None:
                self.timeRemaining -= 1
                if self.timeRemaining <= 0:
                  self.currentTask = None
   
      def busy(self):
            """打印机是否忙?"""
            if self.currentTask is not None:
                return True
            else:
                return False
   
      def startNext(self, newtask):
            """打印新作业!"""
            self.currentTask = newtask
            self.timeRemaining = newtask.getPages() * 60 / self.pagerate
   
   
    class Task:
      def __init__(self, time):
            """ 生成时间戳!"""
            self.timestamp = time
            # 打印页数
            self.pages = random.randint(1, 21)
   
      def getStamp(self):
            return self.timestamp
   
      def getPages(self):
            return self.pages
   
      def waitTime(self, currenttime):
            return currenttime - self.timestamp
   
   
    def newPrintTask():
      num = random.randint(1, 180)
      if num == 180:
            return True
      else:
            return False
   
   
    def simulation(numSeconds, pagesPerMinute):
      #模拟打印,传入打印机使用时间,打印模式(一分钟多少页)
      labprinter = Printer(pagesPerMinute)
      printQueue = Queue()
      waitingtimes = []
      # 时间流逝
      for currentSecond in range(numSeconds):
            if newPrintTask():
                task = Task(currentSecond)
                printQueue.enqueue(task)
   
            if (not labprinter.busy()) and (not printQueue.isEmpty()):
                nexttask = printQueue.dequeue()
                waitingtimes.append(nexttask.waitTime(currentSecond))
                labprinter.startNext(nexttask)
   
            labprinter.tick()
   
      averageWait = sum(waitingtimes) / len(waitingtimes)
      print('Average Wait %.2f secs %3d tasks remaining.' % (averageWait, printQueue.size()))
   
   
    for i in range(10):
      simulation(3600, 10)
    # Average Wait 12.94 secs   0 tasks remaining.
    # Average Wait 14.82 secs   2 tasks remaining.
    # Average Wait 10.75 secs   0 tasks remaining.
    # Average Wait 13.50 secs   0 tasks remaining.
    # Average Wait 37.04 secs   0 tasks remaining.
    # Average Wait 30.11 secs   0 tasks remaining.
    # Average Wait 23.12 secs   0 tasks remaining.
    # Average Wait 33.46 secs   0 tasks remaining.
    # Average Wait 31.94 secs   0 tasks remaining.
    # Average Wait 10.00 secs   1 tasks remaining.
    for i in range(10):
      simulation(3600, 5)
    # Average Wait 156.32 secs   1 tasks remaining.
    # Average Wait 69.20 secs   7 tasks remaining.
    # Average Wait 174.59 secs   2 tasks remaining.
    # Average Wait 44.79 secs   0 tasks remaining.
    # Average Wait 104.91 secs   1 tasks remaining.
    # Average Wait 41.62 secs   0 tasks remaining.
    # Average Wait 96.35 secs   1 tasks remaining.
    # Average Wait 87.29 secs   0 tasks remaining.
    # Average Wait 211.71 secs   0 tasks remaining.
    # Average Wait 199.82 secs   2 tasks remaining.

通过两种情况模拟仿真结果的分析,我们认识到如果有那么多学生要拿着打印好的程序源赶去上课的话,那么就必须得牺牲打印质量,提高打印速度。

模拟系统对现实的仿真,在不耗费现实资源的情况下,有时候真实的实现是无法进行的,可以以不同的设定,反复多次模拟,来帮助我们决策。

四、什么是双端队列(Deque)?

☆ 双端队列:是一种有次序的数据集,跟队列相似,其两端可以称作“首”“尾”端,但deque中数据项既可以从 队首加入,也可以从队尾加入;数据项也可以从两端移除。

      某种意义上来说,双端队列集成了栈和队列的能力。

双端队列并不具有内在的LIFO 或者 FIFO特性,若用双端队列来模拟栈或者队列,那么需要由使用者自行维护操作的一致性。



☆ 抽象数据类型Deque

Deque定义的操作如下:

      Deque(): 创建一个空双端队列

      addFront(item): 将item加入队首

      addRear(item): 将item加入队尾

      removeFront(): 从队首移除数据项,返回值为移除的数据项

      removeRear(): 从队尾移除数据项,返回值为移除的数据项

      isEmpty(): 返回deque是否为空

      size(): 返回deque中包含数据项的个数

      采用List实现,List 作为deque的尾端, List[-1] 作为deque的首端

      操作复杂度:addFront/removeFront: O(1) ,addRear/removeRear: O(n)

    class Deque():
      def __init__(self):
            self.items = []
            
      def isEmpty(self):
            return self.items == []
      
      def addFront(self, item):
            self.items.append(item)
            
      def addRear(self, item):
            self.items.insert(0, item)
         
      def removeFront(self):
            return self.items.pop()
      
      def removeRear(self):
            return self.items.pop(0)
      
      def size(self):
            return len(self.items)

☆ 回文词判定 (回文数同理)

回文词指的是 正读和反读都一样的词, 如 radar, madam,toot

中文: 上海自来水来自海上,山东落花生花落东山

★ 用双端队列很容易解决回文词问题

      先将需要判定的词语从队尾加入deque,再从两端同时移除字符判定是否相同,直到deque中剩下 0 或 1 个字符

    # 回文词判断
    def palchecker(aString):
      chardeque = Deque()
      for ch in aString:
            chardeque.addRear(ch)
      stillEqual = True
      while chardeque.size() > 1 and stillEqual:
            first = chardeque.removeFront()
            last = chardeque.removeRear()
            if first != last:
                stillEqual = False
      return stillEqual
   
    print(palchecker("sdjfhsdjgh"))
    print(palchecker('radar'))
    print(palchecker(('123321')))
    # False
    # True
    # True

木头127 发表于 2022-5-23 12:48

不知道为什么,部分代码显示有问题,没有在代码块里,请见谅。

EragonPaladin1 发表于 2022-5-23 12:57

虽然咱现在没有在学代码了,不过还是可以从上面的流程图中,看出楼主的思路整理的很清晰,想问一下楼主在写代码的时候,是先画思维导图,再写代码,还是在脑中构思一遍,然后直接上手写代码

木头127 发表于 2022-5-23 13:06

EragonPaladin1 发表于 2022-5-23 12:57
虽然咱现在没有在学代码了,不过还是可以从上面的流程图中,看出楼主的思路整理的很清晰,想问一下楼主在写 ...

我是跟着课程学的,在写代码的时候,是先思考问题,把想到的关键点写出来,整理流程,最后再进行代码书写。

zousifang66 发表于 2022-5-23 14:10

请问楼主看的是什么课程呢

木头127 发表于 2022-5-23 14:20

zousifang66 发表于 2022-5-23 14:10
请问楼主看的是什么课程呢

https://www.bilibili.com/video/BV1R541127md/?spm_id_from=333.788
B站,北京大学陈斌老师的Python课程,主讲数据结构和算法,在听之前,建议先听听Python基础语法课程,最好听到 函数和类之后。
页: [1]
查看完整版本: Python学习日记:队列抽象类型及其实现,幸运的基督徒、热土豆问题、回文词问题