首页 > 代码库 > BZOJ 3827: [Poi2014]Around the world

BZOJ 3827: [Poi2014]Around the world

Sol

并查集.

一个点所能到达的最远是单调不降的.然后将链延长到两倍,预处理出每个点到达的最远点,然后倒着计算深度.

再然后一直跳,跳到>=x+n的点,因为跳到的点都能到最终的点,并且不影响后面的答案.

Code

/**************************************************************    Problem: 3827    User: BeiYu    Language: C++    Result: Accepted    Time:70156 ms    Memory:20820 kb****************************************************************/ #include<cstdio>#include<cstring>#include<iostream> using namespace std; const int N = 1000005;const int INF = 0x3fffffff; int n,m,s,lim,ans;int a[N],nxt[N<<1],d[N<<1]; inline int in(int x=0,char ch=getchar()){ while(ch>‘9‘||ch<‘0‘) ch=getchar();    while(ch>=‘0‘&&ch<=‘9‘) x=(x<<3)+(x<<1)+ch-‘0‘,ch=getchar();return x; } int DFS(int x){    int ff=x;    for(;ff<x+n;ff=nxt[ff]);    for(int u=x,v=nxt[x];v!=ff;u=v,v=nxt[v]) nxt[u]=ff;    return d[x]-d[ff];}int main(){//  freopen("in.in","r",stdin);    n=in(),s=in(),m=n<<1|1;    for(int i=1;i<=n;i++) a[i]=in(),lim=max(lim,a[i]);    for(int i=1,x;i<=s;i++){        x=in();ans=INF;        if(x<lim){ puts("NIE");continue; }        for(int u=1,v=1,t=0;u<=m;u++){            while(v<m&&t+a[v>n?v-n:v]<=x)            t+=a[v>n?v-n:v],v++;            nxt[u]=v,t-=a[u>n?u-n:u];        }        for(int i=m;i;i--) d[i]=d[nxt[i]]+1;//      for(int i=1;i<=m;i++) cout<<nxt[i]<<" ";cout<<endl;//      for(int i=1;i<=m;i++) cout<<d[i]<<" ";cout<<endl;                 for(int i=1;i<=n;i++) ans=min(ans,DFS(i));        cout<<ans<<endl;    }return 0;}

  

BZOJ 3827: [Poi2014]Around the world