问题描述:
约瑟夫环问题的原来描述为,设有编号为1,2,……,n的n(n>0)个人围成一个圈,从第1个人开始报数,报到m时停止报数,报m的人出圈,再从他的下一个人起重新报数,报到m时停止报数,报m的出圈,……,如此下去,直到剩下一个人。
利用数学推导,如果能得出一个通式,就可以利用递归、循环等手段解决。下面给出推导的过程:
(1)第一个被删除的数为 (m - 1) % n。
(2)假设第二轮的开始数字为k,那么这n - 1个数构成的约瑟夫环为k, k + 1, k + 2, k +3, .....,k - 3, k - 2。做一个简单的映射。
k -----> 0k+1 ------> 1
k+2 ------> 2
...
...
k-2 ------> n-2
这是一个n -1个人的问题,如果能从n - 1个人问题的解推出 n 个人问题的解,从而得到一个递推公式,那么问题就解决了。假如我们已经知道了n -1个人时,最后胜利者的编号为x,利用映射关系逆推,就可以得出n个人时,胜利者的编号为 (x + k) % n。其中k等于m % n。代入(x + k) % n <=> (x + (m % n))%n <=> (x%n + (m%n)%n)%n <=> (x%n+m%n)%n <=> (x+m)%n
(3)第二个被删除的数为(m - 1) % (n - 1)。
(4)假设第三轮的开始数字为o,那么这n - 2个数构成的约瑟夫环为o, o + 1, o + 2,......o - 3, o - 2.。继续做映射。
o -----> 0
o+1 ------> 1
o+2 ------> 2
...
...
o-2 ------> n-3
这是一个n - 2个人的问题。假设最后的胜利者为y,那么n -1个人时,胜利者为 (y + o) % (n -1 ),其中o等于m % (n -1 )。代入可得 (y+m) % (n-1)要得到n - 1个人问题的解,只需得到n - 2个人问题的解,倒推下去。只有一个人时,胜利者就是编号0。下面给出递推式:
f [1] = 0;
f [ i ] = ( f [i -1] + m) % i; (i>1)
代码:
#include<stdio.h>
main()
{ int i,N,x,m,s;
scanf("%d",&N);
while(N--)
{ s=0;
scanf("%d%d",&m,&x);
for(i=2;i<=m;i++)
s=(s+x)%i;
printf("%d\n",s+1);
}
}
小珂的游戏
-
描述
-
假设有2k个人围着一个圆桌坐着,前k个是好人,后k个是坏人 。现在开始,每m个人踢掉一个,比如有6个人,m=5,那么,被踢掉的人依次是5,4,6,2,3,1。现在要求,在踢掉第一个好人前,必需把所有的坏人踢掉,问,给定一个k,求满足这个要求的最小的m,现在希望你写一个程序,快速的帮助小珂,计算出来这个m。
-
输入
- 每行一个整数k(k<15),0表示输入结束.总测试数据的组数不多于200. 输出
- 各个组对应的最小的m,换行结束。 样例输入
-
3 4 0
样例输出
-
5 30
算法:
前半部分和后半部分,我们不需要知道当前出去的是哪一个,只要知道它的位置在前半部分还是后半部分。
于是如果当前出去的是n,n在后半部分,则下一次剩余数目为rest,步长m则下次出去的是否在后半部分可以用 (n-1+m-1)%rest+1是否在后半部分判断
eg:
1 2 3 4 5 6 其中m=5:
相当于第一次之后1 2 3 4 5第二次取之后1 2 3 4第三次取之后 1 2 3
代码:--选择网络
#include<stdio.h>
int main()
{
int n,m,rest,now;
int sign[15]={0};
while(~scanf("%d",&n),n)
{
if(sign[n]==0)
{ m=0;
while(1)
{ m++;
now=0;
rest=2*n ;
while(1)
{
now=(now+m-1)%rest+1 ;
if(now>n)
{rest--;now--;}
else break;
}
if(rest == n)
{
sign[n] = m;
break;
}
}
}
printf("%d\n",sign[n]);
}
return 0;
}