zoukankan      html  css  js  c++  java
  • BZOJ4227 城市

    https://www.cnblogs.com/clrs97/p/4681014.html

    A国是一个拥有n个城市的国家,其中城市s是A国的首都。
    A国还有m条道路,每条道路连着两个不同的城市,但是一对城市间可能有多条道路。每一条道路都有它的长度,一条道路的通行时间与一条道路的长度成正比。
    你作为A国的统治者,设计出了一种统计城市重要程度的方法:
    1、一条道路的重要度为:在这条道路不能使用的情况下,到首都s的最短时间会变长的城市的数目。
    2、一个城市的重要度为:以它作为一端的所有道路的重要度的和。
    现在,你知道了A国的道路连接情况,你需要计算出每一个城市的重要度。
    Input
    第一行,两个整数n,m,表示有A国有n个城市及m条道路。
    第2~m+1行,每行三个整数u,v,l,描述了一条道路的两个端点城市及长度。
    第m+2行,一个整数s,表示A国首都的编号
    Output
    n行,每行一个整数,第i行为编号i的城市的重要度。

    Sample Input
    4 4

    1 2 3

    2 3 4

    3 4 5

    4 1 2

    1

    Sample Output
    2

    1

    0

    1

    HINT
    100%的数据,1<=n<=100000,1<=m<=200000,1<=l<=10^8

    SOL:

    首先一遍Dijkstra求出S到每个点的最短路,并建出最短路图。

    那么对于一条边,求在这条边不能使用的情况下,到首都S的最短时间会变长的点的数目,等价于求去掉这条边后在最短路图中不能从S出发到达的点的数目。

    对于边x->y,新建一个点z,然后连边x->z->y,这样只需要计算从S到i上必须经过z的点数。

    注意到最短路图是个DAG,我们可以按照拓扑序来建立必经点树:对于一个点x,它在必经点上的父亲为它所有前驱的lca,用倍增即可支持操作。

    时间复杂度O((n+m)logn)。

    #include<cstdio>
    typedef long long ll;
    const int N=300010,M=400010,K=19;
    const ll inf=1LL<<60;
    int n,m,S,cnt,i,x,deg[N],g[N],v[M],w[M],nxt[M],ed;
    int size[N],sum[N],id[N],h,t,q[N],dep[N],f[N][K],G[N],NXT[N],V[N];
    ll d[N],ans[N];
    struct E{int x,y,w;}a[M>>1];
    inline void read(int&a){char c;while(!(((c=getchar())>='0')&&(c<='9')));a=c-'0';while(((c=getchar())>='0')&&(c<='9'))(a*=10)+=c-'0';}
    inline void addedge(int x,int y,int z){v[++ed]=y;w[ed]=z;nxt[ed]=g[x];g[x]=ed;}
    inline void add(int x,int y){deg[y]++;v[++ed]=y;nxt[ed]=g[x];g[x]=ed;}
    inline void addtree(int x,int y){V[++ed]=y;NXT[ed]=G[x];G[x]=ed;}
    struct PI{
      ll x;int y;
      PI(){}
      PI(ll _x,int _y){x=_x,y=_y;}
      inline PI operator+(PI b){return x<=b.x?PI(x,y):b;}
    }val[262145];
    void build(int x,int a,int b){
      val[x]=PI(inf,a);
      if(a==b)return;
      int mid=(a+b)>>1;
      build(x<<1,a,mid),build(x<<1|1,mid+1,b);
    }
    inline void change(int x,int a,int b,int c,ll d){
      if(a==b){val[x].x=d;return;}
      int mid=(a+b)>>1;
      c<=mid?change(x<<1,a,mid,c,d):change(x<<1|1,mid+1,b,c,d);
      val[x]=val[x<<1]+val[x<<1|1];
    }
    inline int lca(int x,int y){
      int i;
      if(dep[x]<dep[y])i=x,x=y,y=i;
      for(i=K-1;~i;i--)if(dep[f[x][i]]>=dep[y])x=f[x][i];
      if(x==y)return x;
      for(i=K-1;~i;i--)if(f[x][i]!=f[y][i])x=f[x][i],y=f[y][i];
      return f[x][0];
    }
    void dfs(int x){
      for(int i=G[x];i;i=NXT[i])dfs(V[i]),size[x]+=size[V[i]];
      sum[id[x]]=size[x];
    }
    int main(){
      read(n),read(m);
      for(i=1;i<=m;i++){
        read(a[i].x),read(a[i].y),read(a[i].w);
        addedge(a[i].x,a[i].y,a[i].w);
        addedge(a[i].y,a[i].x,a[i].w);
      }
      read(S);
      for(i=1;i<=n;i++)d[i]=inf;
      build(1,1,n),change(1,1,n,S,d[S]=0);
      while(val[1].x<inf)for(change(1,1,n,x=val[1].y,inf),i=g[x];i;i=nxt[i])if(d[x]+w[i]<d[v[i]])change(1,1,n,v[i],d[v[i]]=d[x]+w[i]);
      for(ed=0,i=1;i<=n;i++)g[i]=0,size[i]=d[i]<inf;
      for(cnt=n,i=1;i<=m;i++){
        if(d[a[i].x]+a[i].w==d[a[i].y])id[++cnt]=i,add(a[i].x,cnt),add(cnt,a[i].y);
        if(d[a[i].y]+a[i].w==d[a[i].x])id[++cnt]=i,add(a[i].y,cnt),add(cnt,a[i].x);
      }
      for(cnt++,i=1;i<cnt;i++)if(!deg[i])add(cnt,i);
      q[h=t=1]=cnt,ed=0;
      while(h<=t){
        x=q[h++];
        if(f[x][0])addtree(f[x][0],x);
        for(dep[x]=dep[f[x][0]]+1,i=1;i<K;i++)f[x][i]=f[f[x][i-1]][i-1];
        for(i=g[x];i;i=nxt[i]){
          if(!f[v[i]][0])f[v[i]][0]=x;else f[v[i]][0]=lca(f[v[i]][0],x);
          if(!(--deg[v[i]]))q[++t]=v[i];
        }
      }
      dfs(cnt);
      for(i=1;i<=m;i++)ans[a[i].x]+=sum[i],ans[a[i].y]+=sum[i];
      for(i=1;i<=n;i++)printf("%lld
    ",ans[i]);
      return 0;
    }
    

      

  • 相关阅读:
    UVA 11997 K Smallest Sums
    POJ 1007 DNA Sorting
    POJ 3669 Meteor Shower
    POJ 2376 Cleaning Shifts
    POJ 3050 Hopscotch
    操作系统第6次实验报告:使用信号量解决进程互斥访问
    操作系统第5次实验报告:内存管理
    操作系统第4次实验报告:文件系统
    操作系统第3次实验报告:管道
    操作系统第2次实验报告:创建进程
  • 原文地址:https://www.cnblogs.com/cutemush/p/12694540.html
Copyright © 2011-2022 走看看