洛谷 4768 LOJ 2718「NOI2018」归程

【题解】

  本题有多种做法,例如可持久化并查集、kruskal重构树等。

  kruskal重构树的做法是这样的:先把边按照海拔h从大到小的顺序排序,然后跑kruskal建立海拔的最大生成树,顺便建kruskal重构树。

  这样建出来的重构树是一个小根堆,也就是说,如果某个节点没有被淹,它的子树内的点都不会被淹,它们可以互相开车到达。

  我们建重构树的时候维护每个节点的子树内的点到1号点的最小距离mn,mn先用dijkstra处理好。

  1 #include<cstdio>
  2 #include<cstring>
  3 #include<algorithm>
  4 #define LL long long
  5 #define rg register
  6 #define N 400010
  7 using namespace std;
  8 int T,n,m,q,k,s,tot,cnt,last[N],f[N],pos[N],p[N][20],hei[N];
  9 LL ans,mn[N];
 10 struct edge{
 11     int to,pre,dis;
 12 }e[N<<1];
 13 struct rec{
 14     int u,v,h;
 15 }r[N<<1];
 16 struct heap{
 17     LL d;int p;
 18 }h[N];
 19 inline int read(){
 20     int k=0,f=1; char c=getchar();
 21     while(c<'0'||c>'9')c=='-'&&(f=-1),c=getchar();
 22     while('0'<=c&&c<='9')k=k*10+c-'0',c=getchar();
 23     return k*f;
 24 }
 25 inline void up(int x){
 26     int fa;
 27     while((fa=(x>>1))&&h[fa].d>h[x].d) swap(h[x],h[fa]),swap(pos[h[x].p],pos[h[fa].p]),x=fa;
 28 }
 29 inline void down(int x){
 30     int son;
 31     while((son=(x<<1))<=tot){
 32         if(son<tot&&h[son].d>h[son+1].d) son++;
 33         if(h[son].d<h[x].d) swap(h[x],h[son]),swap(pos[h[x].p],pos[h[son].p]),x=son;
 34         else return; 
 35     }    
 36 }
 37 inline void dijkstra(int x){
 38     for(rg int i=1;i<=(n<<1);i++) mn[i]=4e9;
 39     h[tot=pos[x]=1]=(heap){mn[x]=0,x};
 40     while(tot){
 41         int now=h[1].p; pos[h[tot].p]=1; h[1]=h[tot--]; if(tot) down(1);
 42         for(rg int i=last[now],to;i;i=e[i].pre)
 43         if(mn[to=e[i].to]>mn[now]+e[i].dis){
 44             mn[to]=mn[now]+e[i].dis;
 45             if(!pos[to]) h[pos[to]=++tot]=(heap){mn[to],to};
 46             else h[pos[to]].d=mn[to];
 47             up(pos[to]);
 48         }
 49     }
 50 }
 51 inline int climb(int x,int y){
 52     for(rg int i=19;i>=0;i--) if(hei[p[x][i]]>y) x=p[x][i]; return x;
 53 }
 54 int find(int x){return f[x]==x?x:f[x]=find(f[x]);}
 55 inline bool cmp(rec a,rec b){return a.h>b.h;}
 56 inline void Pre(){
 57     tot=0; cnt=0; ans=0;
 58     memset(last,0,sizeof(last));
 59     memset(pos,0,sizeof(pos));
 60     memset(p,0,sizeof(p));
 61 }
 62 int main(){
 63     freopen("return.in","r",stdin);
 64     freopen("return.out","w",stdout);
 65     T=read();
 66     while(T--){
 67         Pre();
 68         n=read(); m=read();
 69         for(rg int i=1;i<=m;i++){
 70             int u,v,d;
 71             r[i].u=u=read(),r[i].v=v=read(),d=read(),r[i].h=read();
 72             e[++tot]=(edge){v,last[u],d}; last[u]=tot;
 73             e[++tot]=(edge){u,last[v],d}; last[v]=tot;
 74         } 
 75         dijkstra(1);
 76     //    for(rg int i=1;i<=n;i++) printf("%d ",mn[i]); puts("mn");
 77         tot=n;
 78         sort(r+1,r+1+m,cmp);
 79         for(rg int i=1;i<=n;i++) f[i]=i,f[i+n]=i+n;
 80         for(rg int i=1;i<=m;i++){
 81             int u=find(r[i].u),v=find(r[i].v);
 82             if(u!=v){
 83                 hei[++tot]=r[i].h; mn[tot]=min(mn[u],mn[v]);
 84                 f[u]=f[v]=p[u][0]=p[v][0]=tot;
 85                 cnt++;
 86             }
 87             if(cnt==n-1) break;
 88         }
 89         for(rg int j=1;j<20;j++)
 90             for(rg int i=1;i<=tot;i++) p[i][j]=p[p[i][j-1]][j-1];
 91         
 92         q=read(); k=read(); s=read();
 93         while(q--){
 94             int v0=(read()+k*ans-1)%n+1,p0=(read()+k*ans)%(s+1);
 95     //        printf("v0=%d p0=%d\n",v0,p0);
 96             printf("%lld\n",ans=mn[climb(v0,p0)]);
 97         }
 98     }
 99     return 0;
100 } 

 

  那么对于每个出发点为v的询问,我们倍增找到它最早的不被淹的祖先p,答案就是mn[p].

 

posted @ 2018-10-15 07:56  Driver_Lao  阅读(191)  评论(0编辑  收藏  举报