美文网首页
约瑟夫问题 -- python实现

约瑟夫问题 -- python实现

作者: 雷子_ | 来源:发表于2019-12-17 11:29 被阅读0次

问题描述

N个人围成一个圈, 从第一个人开始报数, 报到M的人出圈, 剩下的人继续从1开始报数, 报到M的人出圈;如此往复, 直到所有人出圈.

列表解决

def solution_list(n, m):
    """
    初始化一个长度为n的列表, 默认值为True. 当某个元素出圈时, 将其置为False.
    循环迭代这个列表, 遇到值为False的元素则跳过, 当列表中全为False时表示所有人
    都已出圈.
    """
    # 初始化列表
    people = []
    for _ in range(n):
        people.append(True)

    result = []
    num = 1
    while any(people):
        for index, p in enumerate(people):
            if p:
                if num == m:                  # 出圈操作
                    people[index] = False
                    result.append(index + 1)
                    num = 1
                else:
                    num += 1
    print('-' * 25)
    print(f'\n总人数为{n}, 报数为{m}')
    print(f'约瑟夫序列为:\n {result}\n')
    print('-' * 25)
def solution_list2(n, m):
    """
    这是上面这种思路的另一种解法, 将圈内和圈外表示成0和1.
    这里实现循环迭代的方式我第一次遇到, 记录一下
    """
    people = [0 for _ in range(n)]

    alive = n                           # 剩余人数
    index = 0
    num = 0                             # 计数器, 当index == m时出圈
    result = []

    while alive > 0:
        num += 1 - people[index]        # 每轮到一个人报数, 不论0或1都进行计数
        if num == m:
            result.append(index + 1)    # 出圈
            people[index] = 1           # 将出圈人置为 1
            alive -= 1                  # 剩余人数 - 1
            num = 0                     # 重置计数器

        # 与总人数 n 取余, 可以实现index在 0 ~ count -1之间一直循环, 达到循环迭代的目的
        index = (index + 1) % n

    print('-' * 25)
    print(f'\n总人数为{n}, 报数为{m}')
    print(f'约瑟夫序列为:\n {result}\n')
    print('-' * 25)

循环链表解决

class Node:
    """节点"""
    def __init__(self, value):
        self.data = value
        self.next = None

    def __repr__(self):
        return f'Node: {self.data}'


class CircularLinkedList:
    """循环链表"""
    def __init__(self):
        self.rear = None    # 尾节点

    def is_empty(self):
        return self.rear is None

    def append(self, elem):
        """尾插法"""
        temp = Node(elem)
        if self.rear is None:
            temp.next = temp
            self.rear = temp
        else:
            temp.next = self.rear.next
            self.rear.next = temp
            self.rear = temp


def solution_circular_linked_list(n, m):
    """
    通过循环链表解决, 每次出圈弹出该节点
    """
    clist = CircularLinkedList()
    for i in range(n):
        clist.append(i + 1)

    result = []
    pre = clist.rear                    # 当前节点的上一个节点
    cur = clist.rear.next               # 当前节点
    num = 0                             # 计数器

    while cur.next is not cur:
        num += 1
        if num == m:                    # 出圈
            result.append(cur.data)
            pre.next = cur.next         # 弹出当前节点
            num = 0                     # 重置计数器
        else:
            pre = pre.next
        cur = cur.next
    result.append(cur.data)

    print('-' * 25)
    print(f'\n总人数为{n}, 报数为{m}')
    print(f'约瑟夫序列为:\n {result}\n')
    print('-' * 25)

参考:
经典算法--约瑟夫环问题的三种解法
百度百科

相关文章

  • 约瑟夫问题 -- python实现

    问题描述 N个人围成一个圈, 从第一个人开始报数, 报到M的人出圈, 剩下的人继续从1开始报数, 报到M的人出圈;...

  • 算法学习 - Python(持续更新)

    二分查找法 - 递归实现 约瑟夫环问题 汉诺塔问题(递归) 旋转列表(Python):列表原地修改,参考Leetc...

  • 队列实现约瑟夫问题

    约瑟夫问题传说犹太人反叛罗马人,落到困境,约瑟夫和39 人决定殉难,坐成一圈儿,报数1~7,报到7的 人由旁边杀死...

  • php实现约瑟夫环问题

    题目:一群猴子排成一圈,按1,2,…,n依次编号。然后从第1只开始数,数到第m只,把它踢出圈,从它后面再开始数, ...

  • Dart实现约瑟夫环问题

    来源: 最近逛博客,看到有人说面试是遇到问约瑟夫环问题。。。 “约瑟夫”是谁?约瑟夫环又是什么鬼?于是不耻下问(不...

  • python之约瑟夫环问题

    Josephu环:30个人(15个教徒和15个非教徒)坐船出海 船坏 需要把15个人扔到海里 其他人才能幸存 围成...

  • 2019-05-05 约瑟夫问题STL vector解法

    利用STL 标准模板库中的vector或者list实现约瑟夫问题

  • python约瑟夫问题-最全面,经典

    1.需求---- 经典约瑟夫问题 首先,我们需要知道什么是约瑟夫问题?即设有n个人围成一圈,现从第m个人开始报数,...

  • 约瑟夫问题

    今天看了一下约瑟夫问题,嗯,感觉自己智商欠费:( 还是来总结下好啦~ 问题 约瑟夫是犹太军队的一个将军,在反...

  • 约瑟夫问题

    题目:0,1,...,n-1这n个数字排成一个圆圈,从数字0开始每次从这个圆圈里删除第m个数字。求出这个圆圈里剩下...

网友评论

      本文标题:约瑟夫问题 -- python实现

      本文链接:https://www.haomeiwen.com/subject/gzlsnctx.html