约瑟夫环(51nod)

先理解下什么是约瑟夫问题: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)的值 


对于这题

1073 约瑟夫环 

基准时间限制: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;

}

 

 

 

 

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值