poj2886

链接:点击打开链接

题意:有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;
}


 

你可能感兴趣的:(poj2886)