题目:有N个人,编号为1~N,按顺时针围成一个圈,每数m个人,就将这个人从圈中消除。(N<=1e12)
思路:f(1)=0;f(i)=[f(i-1)+m]%i 这里n非常大。
1:当m=1的时候接就是最后的那个人了
2: f(i-1)+m<i 满足条件时,算一下可以跳多少次还在满足条件下,即 f(i-1)+m*num<i-1+num 求满足条件的最大num
如果加上起始位置的话对最后的(ans+s)%n即可。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=1e5+10;
ll n,m,dp[maxn];
int main()
{
while(~scanf("%lld%lld",&n,&m))
{
/*
dp[1]=0;
for(ll i=2;i<=n;i++)
dp[i]=(dp[i-1]+m)%i;
printf("%lld\n",dp[n]+1);
*/
if(m==1)
{
printf("%lld\n",n);
continue;
}
ll ans=0,i=2;
while(i<=n)
{
if(ans+m<i)
{
ll num=(i-ans-1)/(m-1);
if((i-ans-1)%(m-1)==0)
num--;
if(i+num>n)
{
ans=(ans+(n-i+1)*m)%n;
break;
}
else
{
i+=num;
ans=(ans+m*num)%i;
}
}
else
{
ans=(ans+m)%i;
i++;
}
}
printf("%lld\n",ans+1ll);
}
return 0;
}