Py递归算法解“约瑟夫环”的一种变形问题


【点击此处跳转笔记正文】

Python 官网:https://www.python.org/


  • Free:大咖免费“圣经”教程《 python 完全自学教程》,不仅仅是基础那么简单……

  • My CSDN主页、My HOT博、My Python 学习个人备忘录
  • 好文力荐、 老齐教室

  自学并不是什么神秘的东西,一个人一辈子自学的时间总是比在学校学习的时间长,没有老师的时候总是比有老师的时候多。
            —— 华罗庚


等风来,不如追风去……

这是我参加“14天阅读挑战赛”第一周第一篇

点击查看“14天阅读挑战赛”详情,《趣学算法第二版》1.1 一棋盘的麦子。

  跟着《趣学算法第二版》作者陈小玉老师学算法。昨晚一场直播,如醍醐灌顶,豁然开朗。以前人都是边写代码边“规划”数据结构,经常有大段代码推倒重建的尴尬。用学到的方法,先捋需求(题意),构建数据结构,根据数据结构特点选取算法,画算法图解,优化算法,尽可能做到最优。最后才是算法语言实现。所谓“算法”,即是用自然语言写伪代码的过程。最后,用所长的语言实现算法其实最最简单的一环。
  我刚好看到py学习群有贴出“约瑟夫环”的一种变形问题,感觉好玩儿,试着用昨天学习到的方法,用递归和数组的方式搞“算法”。结果,是我写Python代码感觉最轻松的一次——水到渠成的满满成就感。



Py递归算法
解“约瑟夫环”
(“约瑟夫环”的一种变形问题)


题目:

约瑟夫环的一种变形。我们考虑的问题是开始时有n个人,记为1到n,站成一个圆圈。每一步,每第2个人将被去除,直到只剩下一个人为止。我们把剩下的人记为J(n),用程序求出1≤n≤16的J(n)(“约瑟夫环”点此跳转百科词条)

思路解析:
  根据题意,要把人依序站队,数据结构选择数组(list)最为合适,用整数1~n来标识队列中人的初始站队位置。要让队列像贪吃蛇一样首尾相接成为圆圈,依list特性借助了list的pop和append方法,实现去除人和留下的人接在队列后面,数数点不动圆圈队列转圈圈。
  题目把剩下的人记为J(n),虽然感觉有点怪异,还是把函数名设定为j(n)(小写字母才符Python函数命名规范),n为队列总人数。

递归去除人函数代码


def j(n):
    ''' 解“约瑟夫环”一种变形问题。'''
    persons = list(range(1, n+1))
    
    def cook(persons):
        ''' 递归去除人函数。'''
        # print(persons) # 调试用语句。
        if len(persons) == 1:
            return persons[0] # 队列只剩下一人,停止递归。

        persons.append(persons.pop(0)) # 留下的人站到队属,实现圆环站队。
        print(persons.pop(0), end=', ') # 打印去除人的位置。
        # print(persons) # 调试用语句。

        return cook(persons) # 队列大于一人,递归去除。

函数精简代码:persons人总数为n的list生成,必不可少,终止递归的if语句必有,留下的人append及去除人的pop语句是函数干活的主力,递归调用函数自己的语句是“循环”的保障。算起来,也就五条语句


def j(n):
    ''' 解“约瑟夫环”一种变形问题。'''
    persons = list(range(1, n+1))
    
    def cook(persons):
        ''' 递归去除人函数。'''
        if len(persons) == 1:
            return persons[0] # 队列只剩下一人,停止递归。

        persons.append(persons.pop(0)) # 留下的人站到队属,实现圆环站队。
        print(persons.pop(0), end=', ') # 打印去除人的位置。

        return cook(persons) # 队列大于一人,递归去除。


回页首

代码运行效果截屏
Py递归算法解“约瑟夫环”的一种变形问题_第1张图片


回页首

Pycolor完整源码(点此跳过源码)

#!/sur/bin/nve python
# coding: utf-8

'''

Author:梦幻精灵_cq

date:2022-10-18

'''

from  time import localtime


# 打印颜色设置字符。
blue = '\033[34;5m'
green = '\033[92;5m'
offall = '\033[0m'

def j(n):
    ''' 解“约瑟夫环”一种变形问题。'''
    persons = list(range(1, n+1))
    
    def cook(persons):
        ''' 递归去除人函数。'''
        # print(persons) # 调试用语句。
        if len(persons) == 1:
            return persons[0] # 队列只剩下一人,停止递归。

        persons.append(persons.pop(0)) # 留下的人站到队属,实现圆环站队。
        print(persons.pop(0), end=', ') # 打印去除人的位置。
        # print(persons) # 调试用语句。

        return cook(persons) # 队列大于一人,递归去除。

    return cook(persons)

if __name__ == '__main__':

    for i in range(1, 17):
        print('\n去除人位置顺序:', end='')
        print(f"\n当队列为{blue}{i:2}{offall}人时,约瑟夫环最后剩下的是第{green}{j(i):2}{offall}人。")

    print()
    y, M, d, h, m, s = localtime()[:6]
    input(f"{'~'*14} {blue}{y}-{M:02}-{d:02}{offall} {green}{h:02}:{m:02}:{s:02}{offall} {'~'*15}")


回页首

__上一篇:__ 随机颜色彩色提示字符串展示

__下一篇:__ 

我的HOT博:

    • New:给定字符串提取姓名(字符串、list、re“零宽断言”)(1051阅读)
    • New:我的 Python.color() (Python 色彩打印控制)(1125阅读)
    • New:python清屏(1290阅读)
    • 回车符、换行符和回车换行符(1322阅读)
    • Linux 脚本文件第一行的特殊注释符(井号和感叹号组合)的含义(1171阅读)
    • pandas 数据类型之 Series(1224阅读)
    • 聊天消息敏感词屏蔽系统(字符串替换 str.replace(str1, *) )(1270阅读)
    • 练习:银行复利计算(用 for 循环解一道初中小题)(1188阅读)
    • pandas 数据类型之 DataFrame(2136阅读)
    • :班里有人和我同生日难吗?(蒙特卡洛随机模拟法)(2180阅读)
    • Python字符串居中显示(2359阅读)
    • 练习:求偶数和、阈值分割和求差( list 对象的两个基础小题)(1665阅读)
    • 用 pandas 解一道小题(2007阅读)
    • 可迭代对象和四个函数(1083阅读)
    • “快乐数”判断(1252阅读)
    • 罗马数字转换器(构造元素取模)(2159阅读)
    • Hot:罗马数字(转换器|罗生成器)(4750阅读)
    • Hot:让QQ群昵称色变的代码(36654阅读)
    • Hot:斐波那契数列(递归| for )(4071阅读)
    • 柱状图中最大矩形(1663阅读)
    • 排序数组元素的重复起止(1258阅读)
    • 电话拨号键盘字母组合(1402阅读)
    • 密码强度检测器(1986阅读)
    • 求列表平衡点(1837阅读)
    • Hot: 字符串统计(4308阅读)
    • Hot:尼姆游戏(聪明版首发)(3493阅读)尼姆游戏(优化版)(1175阅读)
    • 推荐条件 点阅破千

      回目录


      老齐漫画头像

      精品文章:

      • 好文力荐:《python 完全自学教程》齐伟书稿免费连载
      • OPP三大特性:封装中的property
      • 通过内置对象理解python'
      • 正则表达式
      • python中“*”的作用
      • Python 完全自学手册
      • 海象运算符
      • Python中的 `!=`与`is not`不同
      • 学习编程的正确方法

      来源:老齐教室


      回目录

      Python 入门指南【Python 3.6.3】


      好文力荐:

      • 全栈领域优质创作者——寒佬(还是国内某高校学生)好文:《非技术文—关于英语和如何正确的提问》,“英语”和“会提问”是学习的两大利器。

      • 【8大编程语言的适用领域】先别着急选语言学编程,先看它们能干嘛

      • 靠谱程序员的好习惯


      CSDN实用技巧博文:

      • 8个好用到爆的Python实用技巧
      • python忽略警告
      • Python代码编写规范
      • Python的docstring规范(说明文档的规范写法)

    你可能感兴趣的:(笔记,python,算法)