链接:点击打开链接
题意:有n个人围成一个圈,第一次是顺时针第k个人出列,当这个人出列后,下个出队的是从这个人开始数第num[i]个人,如果num[i]<0则为逆时针数,否则顺时针数。重复这个过程直到全部出列,输出出列的人的序号中,序号的约数最多的那个人的名称及约数个数
题意:
#include<stdio.h> #include<string.h> #include<iostream> const int SIZE=500005; int tree[SIZE<<2]; using namespace std; int a[40] = {1,2,4,6,12,24,36,48,60,120,180,240,360,720,840,1260,1680,2520,5040,7560,10080,15120,20160,25200,27720,45360,50400,55440,83160,110880,166320,221760,277200,332640,498960,500001}; int b[40] = {1,2,3,4,6,8,9,10,12,16,18,20,24,30,32,36,40,48,60,64,72,80,84,90,96,100,108,120,128,144,160,168,180,192,200,1314521}; struct node{ char str[50]; int num; }; node s[SIZE]; void build(int l,int r,int rt){ int m; if(l==r){ tree[rt]=1; return; } m=(l+r)>>1; build(l,m,rt<<1); build(m+1,r,rt<<1|1); tree[rt]=tree[rt<<1]+tree[rt<<1|1]; } int query(int p,int l,int r,int rt){ int m,ans; if(l==r){ tree[rt]=0; return l; } m=(l+r)>>1; if(p<=tree[rt<<1]) ans=query(p,l,m,rt<<1); else ans=query(p-tree[rt<<1],m+1,r,rt<<1|1); tree[rt]=tree[rt<<1]+tree[rt<<1|1]; return ans; } int main(){ int n,k,i,ans,tmp,pos,cnt; //因为最大值就是小于n的反素数的最大值,因此 while(scanf("%d%d",&n,&k)!=EOF){ //先将反素数的表打出来 for(i=1;i<=n;i++) scanf("%s%d",s[i].str,&s[i].num); for(i=0;a[i]<=n;i++) tmp=a[i],ans=b[i]; build(1,n,1); cnt=n; while(tmp--){ n--; pos=query(k,1,cnt,1); //用线段树求出是当前序列的第几个元素 if(n==0) //线段树部分跟hdu2795很相似 break; if(s[pos].num>=0) //跟据当前的k推出下一个k k=(k+s[pos].num-2)%n+1; else k=((k+s[pos].num-1)%n+n)%n+1; } printf("%s %d\n",s[pos].str,ans); } return 0; }