===================================================================
据说著名犹太历史学家Josephus有过以下的故事:在罗马人占领乔塔帕特后,39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人抓到,于是决定了一个自杀方式,41个人排成一个圆圈,由第1个人开始报数,每报数到第3人该人就必须自杀,然后再由下一个重新报数,直到所有人都自杀身亡为止。然而Josephus 和他的朋友并不想遵从。首先从一个人开始,越过k-2个人(因为第一个人已经被越过),并杀掉第k个人。接着,再越过k-1个人,并杀掉第k个人。这个过程沿着圆圈一直进行,直到最终只剩下一个人留下,这个人就可以继续活着。问题是,给定了和,一开始要站在什么地方才能避免被处决。Josephus要他的朋友先假装遵从,他将朋友与自己安排在第16个与第31个位置,于是逃过了这场死亡游戏。
17世纪的法国数学家加斯帕在《数目的游戏问题》中讲了这样一个故事:15个教徒和15 个非教徒在深海上遇险,必须将一半的人投入海中,其余的人才能幸免于难,于是想了一个办法:30个人围成一圆圈,从第一个人开始依次报数,每数到第九个人就将他扔入大海,如此循环进行直到仅余15个人为止。问怎样排法,才能使每次投入大海的都是非教徒。
===================================================================
我们学校最近两次的蓝桥杯校选拔赛都出现过约瑟夫问题,怎么说呢,这种类型的题目虽然不难,而且当时我也做出来了,但是需要时间,脑海中没有一个清晰的框架,做是做出来了,也是比较模糊的那种一步步模拟出来的,可能稍微变一下型又要想一会时间了。
刚好今天偶遇这条题,就顺便记录一下,也总结出了一个模板,下次遇到直接套用就完事,省时省力。
=================================================================
环形圈可以将其转换为一个一维列表来存储。
先以问题一为例,如果跟着题目思路走的话应该就是从起始位置出发,定义一个指针,指向此时走到的位置。开始往后数,每数到3就把一个元素给踢走,反复这么做。当遍历完列表最后一个元素后, 指针又返回第一个元素,重新遍历,重复上面动作,直到选出最后所需元素为止。
上面说的方法,是比较容易想到的方法,我称其为“完全模拟法”,就是完全跟着题目思路走一步步模拟的。这种思路的优点就是比较容易想到,缺点就是容易出错,每弹出一个元素,列表长度会改变,相应的往后元素的下标值也会发生改变,这时要十分注意,稍微不小心就会出错。
那么我们就想想有没有什么方法能够避免这种情况。
方法当然是有的,而且很多,我就以其中一种方法为例讲解一下。
=================================================================
我们把环形圈当成一个列表队列,每次报数当做是队列元素的出队入队操作,我们定义一个计数器,记录此时报到第几个元素,每次只对列表首个元素进行判断,如果报到3,就将它踢出队列,否则,将它放到队尾。不断循环出队入队操作,直到队中元素个数满足要求为止。
这么做其实跟“完全模拟法”的思路差不多的,都没有改变元素的相对顺序,但这么做能够减少因为列表长度变换而引起的麻烦。
如果思路不懂的同学可以手写模拟一下出队入队的过程。
还有一种高阶的做法是,公式法,找出其中的数学规律进行解答,时间空间都能够优化到极致。可参考这篇文章。
=================================================================
这里我总结出了一个模板,可应对总人数不同,间隔数不同,剩余人数不同的情况。 但空间时间效率优化的不够好,面对大额数时会显得比较鸡肋,因此,模板适用于竞赛填空题。
n,m,k= map(int,input().split()) # n,m,k分别表示 总人数,间隔数,剩余人数
nums = [i for i in range(1,n+1)] # 人数列表
cnt = 1 # 计数器
while len(nums) > m: # 如果列表长度大于剩余人数,则循环操作列表
num = nums.pop(0) # 弹出一个元素
if cnt < k: # 如果没有踩雷那么就将其放入队尾
nums.append(num)
cnt += 1 # 计数器加1
else: # 踩雷就不放回队尾了
一、Python所有方向的学习路线
Python所有方向的技术点做的整理,形成各个领域的知识点汇总,它的用处就在于,你可以按照下面的知识点去找对应的学习资源,保证自己学得较为全面。
二、Python必备开发工具
工具都帮大家整理好了,安装就可直接上手!
三、最新Python学习笔记
当我学到一定基础,有自己的理解能力的时候,会去阅读一些前辈整理的书籍或者手写的笔记资料,这些笔记详细记载了他们对一些技术点的理解,这些理解是比较独到,可以学到不一样的思路。
四、Python视频合集
观看全面零基础学习视频,看视频学习是最快捷也是最有效果的方式,跟着视频中老师的思路,从基础到深入,还是很容易入门的。
五、实战案例
纸上得来终觉浅,要学会跟着视频一起敲,要动手实操,才能将自己的所学运用到实际当中去,这时候可以搞点实战案例来学习。
六、面试宝典
简历模板![在这里插入图片描述](https://img-blog.csdnimg.cn/646863996ac44da8af500c049bb72fbd.png#pic_center)
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!