约瑟夫环的链表解法和数学解法

105 篇文章 0 订阅

   约瑟夫环(Josephus)问题是由古罗马的史学家约瑟夫(Josephus)提出的,他参加并记录了公元66—70年犹太人反抗罗马的起义。约瑟夫作为一个将军,设法守住了裘达伯特城达47天之久,在城市沦陷之后,他和40名死硬的将士在附近的一个洞穴中避难。在那里,这些叛乱者表决说“要投降毋宁死”。于是,约瑟夫建议每个人轮流杀死他旁边的人,而这个顺序是由抽签决定的。约瑟夫有预谋地抓到了最后一签,并且,作为洞穴中的两个幸存者之一,他说服了他原先的牺牲品一起投降了罗马。

约瑟夫环问题的具体描述是:设有编号为1,2,……,n的n(n>0)个人围成一个圈,从第1个人开始报数,报到m时停止报数,报m的人出圈,再从他的下一个人起重新报数,报到m时停止报数,报m的出圈,……,如此下去,直到所有人全部出圈为止。当任意给定n和m后,设计算法求n个人出圈的次序。 

 

解法一:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include<stdio.h>  
  2. #include<stdlib.h>  
  3.    
  4. typedef struct Node  
  5. {  
  6.        int data;  
  7.        struct Node *next;  
  8. } Node;  
  9.    
  10. /** 
  11.  * @功能约瑟夫环 
  12.  * @参数 total:总人数 
  13.  * @参数 from:第一个报数的人 
  14.  * @参数 count:出列者喊到的数 
  15.  * @更新 2013-12-5 
  16.  */  
  17. void JOSEPHUS(int total, int from, int count)  
  18. {  
  19.     // pcur为当前结点,pre为辅助结点,指向pcur的前驱结点,head为头节点  
  20.     Node *pcur, *pre, *head;  
  21.     head = NULL;  
  22.     int i;  
  23.      
  24.     // 建立循环链表  
  25.     for(i = 1; i <= total; i++)  
  26.     {  
  27.         pcur = (Node *)malloc(sizeof(Node));  
  28.         pcur->data = i;  
  29.         if(NULL ==head)  
  30.          {  
  31.            head = pcur;  
  32.          }  
  33.         else  
  34.          {  
  35.             pre->next = pcur;  
  36.          }  
  37.          pre = pcur;  
  38.     }  
  39.     pcur->next = head;      // 尾节点连到头结点,使整个链表循环起来  
  40.     pcur = head;            // 使pcur指向头节点  
  41.      
  42.     // 把当前指针移动到第一个报数的人,即第k位的下一位  
  43.     for(i = 1; i < from; i++)  
  44.     {  
  45.         pre = pcur;  
  46.         pcur = pcur->next;  
  47.     }  
  48.      
  49.     // 循环地删除队列结点,每隔m-1个结点删一个  
  50.     while(pcur->next != pcur)  
  51.     {  
  52.         for(i = 1; i < count; i++)  
  53.         {  
  54.             pre = pcur;  
  55.             pcur = pcur->next;  
  56.         }  
  57.         pre->next =pcur->next;  
  58.         printf("deletenumber: %d\n", pcur->data);  
  59.         free(pcur);  
  60.         pcur = pre->next;  
  61.     }  
  62.      
  63.     printf("Thelast one is No.%d\n", pcur->data);  
  64. }  
  65.    
  66. int main()  
  67. {  
  68.        // 总共有13人,从第1位开始报数,每隔两位踢出1个。  
  69.        JOSEPHUS(13,1,3);  
  70. }  

运行结果:

delete number: 3

delete number: 6

delete number: 9

delete number: 12

delete number: 2

delete number: 7

delete number: 11

delete number: 4

delete number: 10

delete number: 5

delete number: 1

delete number: 8

Thelast one is No.13

 

解法二:

假设下标从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

 

源程序如下:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  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


zz  http://blog.csdn.net/haishu_zheng/article/details/17220429

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值