一:问题
1.问题描述
有n个小朋友围成一圈玩游戏,小朋友从1至n编号,2号小朋友坐在1号小朋友的顺时针方向,3号小朋友坐在2号小朋友的顺时针方向,……,1号小朋友坐在n号小朋友的顺时针方向。
游戏开始,从1号小朋友开始顺时针报数,接下来每个小朋友的报数是上一个小朋友报的数加1。若一个小朋友报的数为k的倍数或其末位数(即数的个位)为k,则该小朋友被淘汰出局,不再参加以后的报数。当游戏中只剩下一个小朋友时,该小朋友获胜。
例如,当n=5, k=2时:
1号小朋友报数1;
2号小朋友报数2淘汰;
3号小朋友报数3;
4号小朋友报数4淘汰;
5号小朋友报数5;
1号小朋友报数6淘汰;
3号小朋友报数7;
5号小朋友报数8淘汰;
3号小朋友获胜。
给定n和k,请问最后获胜的小朋友编号为多少?
2.格式
输入格式
输入一行,包括两个整数n和k,意义如题目所述。
输出格式
输出一行,包含一个整数,表示获胜的小朋友编号。
样例输入
5 2
样例输出
3
样例输入
7 3
样例输出
4
3.数据规模和约定
对于所有评测用例,1 ≤ n ≤ 1000,1 ≤ k ≤ 9。
二:理解
这个类似于约瑟夫环,循环报数,不过这个题目是(当报的数是k的倍数,或者个位是k时,就可以将其淘汰;其他的数依然按着顺序报数)
1.实现循环
while(1)
{
for(i=1; i<=n; i++)
{
...
if(count == n) //这是最后一个数,就可以结束循环
break;
}
if(count==n)
break;
}
2.将n个数,用一个数组来标记每个数的状态
int sign[1005]={0}; //标志是否被淘汰
当sign[i] == 1时,这个数就是被淘汰了。
3.当这个数没有被淘汰时,用num来计数;
判断应不应该被淘汰:
if(!sign[i])
{
num++;
int temp = num%10; //看其个位是否为k
if(temp==k || num%k==0) //淘汰
{
sign[i] = 1;
count++;
}
}
三:代码
#include<iostream>
#include<algorithm>
#include<cstdio>
using namespace std;
int sign[1005]={0}; //标志是否被淘汰
int main()
{
int n, k;
cin >> n >> k;
int count = 0; //被淘汰的人数
int num = 0; //报数
int i;
while(1)
{
for(i=1; i<=n; i++)
{
if(!sign[i])
{
num++;
int temp = num%10; //看其个位是否为k
if(temp==k || num%k==0) //淘汰
{
sign[i] = 1;
count++;
}
}
if(count == n) //这是最后一个数,就可以结束循环
break;
}
if(count==n)
break;
}
cout << i <<endl;
return 0;
}
写的第二次代码:
这个题的理解,先报数,判断这个数是不是满足淘汰的要求。
并且,报数的人的编号也要一直更新。
#include<bits/stdc++.h>
#include<iostream>
#include<algorithm>
using namespace std;
int visited[1005] = {0};
int main()
{
int counts = 0; //报的总数
int index = 1; //模仿人数
int n, k;
int i = 0; //淘汰的个数
cin >> n >> k;
while(1)
{
if(!visited[index]) //报数增加
counts++;
//满足要求的,个数增加,标记为已经存在了
if((!(counts%k) || counts%10 == k) && !visited[index])
{
visited[index] = 1;
i++;
}
//更新报数的人
if(index == n)
index = 1;
else
index++;
if(i == n-1)
break;
}
for(int i=1; i<=n; i++)
if(!(visited[i]))
{
cout << i << endl;
break;
}
return 0;
}