约瑟夫环(Josephus)问题是由古罗马的史学家约瑟夫(Josephus)提出的,他参加并记录了公元66—70年犹太人反抗罗马的起义。约瑟夫作为一个将军,设法守住了裘达伯特城达47天之久,在城市沦陷之后,他和40名死硬的将士在附近的一个洞穴中避难。在那里,这些叛乱者表决说“要投降毋宁死”。于是,约瑟夫建议每个人轮流杀死他旁边的人,而这个顺序是由抽签决定的。约瑟夫有预谋地抓到了最后一签,并且,作为洞穴中的两个幸存者之一,他说服了他原先的牺牲品一起投降了罗马。
假设下标从0开始,0,1,2 … m-1共m个人,从第0个开始报数,报到k则此人从环出退出,问最后剩下的一个人的编号是多少?
现在假设m=10
01 2 3 4 5 6 7 8 9 k=3
第一个人出列后的序列为:
01 3 4 5 6 7 8 9
即:
34 5 6 7 8 9 0 1(*)
我们把该式转化为:
01 2 3 4 5 6 7 8 (**)
则你会发现: ((**)+3)%10则转化为(*)式了
也就是说,我们求出9个人中第9次出环的编号,最后进行上面的转换就能得到10个人第10次出环的编号了。这样,9个人中第9次出环(或10个人中第10次出环)的编号,就是最后的结果。
设f(m,k,i)为m个人的环,报数为k,第i个人出环的编号,则f(10,3,10)是我们要的结果
当i=1时, f(m,k,i) = (m+k-1)%m(这里减1是因为从0开始计数)
当i!=1时, f(m,k,i)= ( f(m-1,k,i-1)+k )%m
- #include <stdio.h>
- /**
- * @功能 求约瑟夫环中最后留下来的人
- * @更新 2013-12-5
- */
- int fun(int m,int k,int i)
- {
- if(1 == i)
- {
- return (m + k - 1) % m;
- }
- else
- {
- return (fun(m - 1, k, i - 1) + k) % m;
- }
- }
- int main(int argc, char* argv[])
- {
- for(int i = 1; i <= 13; i++)
- {
- printf("第%d次出环:%d\n", i, fun(13, 3, i));
- }
- return 0;
- }
运行结果:
第1次出环:2
第2次出环:5
第3次出环:8
第4次出环:11
第5次出环:1
第6次出环:6
第7次出环:10
第8次出环:3
第9次出环:9
第10次出环:4
第11次出环:0
第12次出环:7
第13次出环:12
非递归解法,设m=10,k=3
var s=0;
for (var i=2; i<=n; i++)
s=(s+3)%i;
return (s+1);