约瑟夫环的数学解决

  约瑟夫环(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

 


  1. #include <stdio.h>  
  2.    
  3. /** 
  4.  * @功能 求约瑟夫环中最后留下来的人 
  5.  * @更新 2013-12-5 
  6.  */  
  7. int fun(int m,int k,int i)  
  8. {  
  9.        if(1 == i)  
  10.        {  
  11.          return (m + k - 1) % m;  
  12.        }  
  13.        else  
  14.        {  
  15.          return (fun(m - 1, k, i - 1) + k) % m;  
  16.        }  
  17. }  
  18.    
  19. int main(int argc, char* argv[])  
  20. {  
  21.        for(int i = 1; i <= 13; i++)  
  22.        {  
  23.          printf("第%d次出环:%d\n", i, fun(13, 3, i));  
  24.        }  
  25.        return 0;  
  26. }  

运行结果:

第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);

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值