约瑟夫环问题,经典的算法问题。
这个问题说的是一个小故事。在很久以前,有n个犹太人遭到敌人的追击,他们逃到了一个山洞中,大部分的人决定宁愿去死也不要让敌人抓住。他们围成一圈,一个个轮流报数,报到k的人就要当场自杀,然后下一个人从1开始重新数数。而约瑟夫不想自杀,于是他灵机一动,站到了圈的一个位置上,结果其他人一个个都自杀了,最后只剩下约瑟夫一个人还活着,最终他逃出了山洞。那么问题来了,约瑟夫站的是哪个位置呢?
这个问题的实质就是,n个人围成一个圈轮流数数,每次淘汰数到k的那个人,求最后剩下的人是谁。
这个问题有很多解法,但大多都是比较复杂的解决方法,在这里我先介绍一种巧妙的递推方式。先上递推关系式:
F[1]=0;
F[n]=(F[n-1]+k) mod n;
刚看这个公式估计会有点蒙,待我一步步说来。
首先为了编程方便,我们假定n个人的编号为0,1,...,n-1。
然后,F[n]是我们的目标函数,也就是最后剩下那个人的编号。
我们先假设已经知道F[n-1]的值,例如F[n-1]=P,也就是
0,1,...,P-1,P,P+1,...,N-2
那么我们怎么通过这个P来求的所需要的F[n]呢,实际上只需简单绕个弯。
由于n与n-1之间只相差1