先理解下什么是约瑟夫问题:https://blog.csdn.net/tingyun_say/article/details/52343897
以及解决方法:https://blog.csdn.net/wusuopubupt/article/details/18214999
题目:n个人坐成一个环(编号0~n-1),从第1个人开始报数,数到q的人出列,后面的人重新从1开始报数。问最后剩下的人的编号。
上面那篇博客先讲述了对于q=2,n=2^k的特殊情况:
首先看看n=2的情况:显然剩下的是第一个人,也就是0号
如果n=4时:第一轮 去掉了1号和3号,剩下0号和2号,第二轮就剩下0号
如果n=8时:第一轮
0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 |
去掉1,3,5,7号,剩0,2,4,6号,再一轮,剩下0,4号,最后剩下的还是0号
可见对于q=2,n=2^k的情况最后剩下的都是0号
如果n!=2^k时,可以将其写成n=2^k+t;
以9为例,9=2^3 +1;我们可以先去掉一个,那么就剩下8个人,这样就变成了n=2^k的特殊情况
n=9去掉1号后变成了8个人
0 | 2 | 3 | 4 | 5 | 6 | 7 | 8 |
我们将0号放到最后面,然后对其重新编号
2 | 3 | 4 | 5 | 6 | 7 | 8 | 0 |
0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 |
由q=2,n=2^k的特殊情况:可知,对于n==8时,剩下的便是0号,对应的就是n==9时的2号
因此q=2时,对于任意的n,都可以写成n=2^k+t;我们可以先去掉t个人,将其转化为n=2^k的问题,有
J(2^k+t)=2*t; J(n)表示的是n个人最后剩下人的编号,所有人编号按(0~n-1)编号
现在我们来讨论对于任意的n与q
现在以n=10为例,每次将q-1号删掉,然后将0到q-2号放到最后面,然后再重新编号,下面括号里的是未编号前的编号
这样一直将n减小,直到n=1,而n=1是剩下的就是0号,然后通过J(2)与J(1)编号之间的对应关系,就可以找到在J(2)中对应的编号
以此类推,最后就可以求出J(n)中对应的编号,即所求的答案
从上面面的J(n)与J(n-1)可以发现J(n-1)中的编号X,在J(n)中对应的编号X',存在关系:X'=(X+q)%n;
因此也就有J(n)=(J(n-1)+q)%n;
可以知道J(1)=1
则J (2)=(J(1)+q)%2;........J(n-1)=(J(n-2)+q)%(n-1);J(n)=(J(n-1)+q)%n;
最后就得到了J(n);
代码实现:
int res=0;
for(int i=2;i<=n;i++)
res=(res+q)%i;
最后res的值就是J(n)的值
对于这题
基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题
收藏
关注
N个人坐成一个圆环(编号为1 - N),从第1个人开始报数,数到K的人出列,后面的人重新从1开始报数。问最后剩下的人的编号。
例如:N = 3,K = 2。2号先出列,然后是1号,最后剩下的是3号。
Input
2个数N和K,表示N个人,数到K出列。(2 <= N, K <= 10^6)
Output
最后剩下的人的编号
Input示例
3 2
Output示例
3
由于这题的编号是从1到n,故只需将J(n)+1就行了
代码:
#include<cstdio>
#include<iostream>
#include<cmath>
using namespace std;
#define ll long long
int main()
{
int n;
int q;
while(scanf("%d%d",&n,&q)==2)
{
int res=0;
for(int i=2;i<=n;i++)
res=(res+q)%i;
printf("%d\n",res+1);
}
return 0;
}