BZOJ3887 [Usaco2015 Jan]Grass Cownoisseur[缩点]
首先看得出缩点的套路。跑出DAG之后,考虑怎么用逆行条件。首先可以不用,这样只能待原地不动。用的话,考虑在DAG上向后走,必须得逆行到1号点缩点后所在点的前面,才能再走回去。
于是统计从1号点缩点所在点到所有走到的点的最长距离,以及所有可以走到1号的点到1号的最长距离。然后,看在哪里逆行,可以暴力枚举每条边,然后把两边连接的点用预处理好的信息更新答案即可。这个可以使用正/反向跑图加记忆化。
注意一点:这样统计是不会重复统计的,因为如果存在点可以从1号经过,逆行之后又经过这个点,回到1号点,那就有环了,不是DAG。
忽略点($WA\times 1$):1号点也可能在环内,要以缩点之后的代表点来考虑。
1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<algorithm> 5 #include<cmath> 6 #include<queue> 7 #define dbg(x) cerr << #x << " = " << x <<endl 8 #define dbg2(x,y) cerr<< #x <<" = "<< x <<" "<< #y <<" = "<< y <<endl 9 using namespace std; 10 typedef long long ll; 11 typedef double db; 12 typedef pair<int,int> pii; 13 template<typename T>inline T _min(T A,T B){return A<B?A:B;} 14 template<typename T>inline T _max(T A,T B){return A>B?A:B;} 15 template<typename T>inline char MIN(T&A,T B){return A>B?(A=B,1):0;} 16 template<typename T>inline char MAX(T&A,T B){return A<B?(A=B,1):0;} 17 template<typename T>inline void _swap(T&A,T&B){A^=B^=A^=B;} 18 template<typename T>inline T read(T&x){ 19 x=0;int f=0;char c;while(!isdigit(c=getchar()))if(c=='-')f=1; 20 while(isdigit(c))x=x*10+(c&15),c=getchar();return f?x=-x:x; 21 } 22 const int N=1e5+7,INF=0x3f3f3f3f; 23 struct thxorz{int to,nxt;}G[N],dag[N],rdag[N]; 24 int Head[N],dhd[N],rhd[N],tot1,tot2,tot3,frm[N]; 25 int n,m; 26 inline void Addedge(int x,int y){G[++tot1].to=y,G[tot1].nxt=Head[x],Head[x]=tot1;frm[tot1]=x;} 27 inline void AddDAG(int x,int y){dag[++tot2].to=y,dag[tot2].nxt=dhd[x],dhd[x]=tot2;} 28 inline void AddrDAG(int x,int y){rdag[++tot3].to=y,rdag[tot3].nxt=rhd[x],rhd[x]=tot3;} 29 #define y G[j].to 30 int dfn[N],low[N],cnt,instk[N],stk[N],Top,val[N],bel[N]; 31 void tarjan(int x){ 32 dfn[x]=low[x]=++cnt,stk[++Top]=x,instk[x]=1; 33 for(register int j=Head[x];j;j=G[j].nxt){ 34 if(!dfn[y])tarjan(y),MIN(low[x],low[y]); 35 else if(instk[y])MIN(low[x],dfn[y]); 36 } 37 if(dfn[x]==low[x]){int tmp;do instk[tmp=stk[Top--]]=0,bel[tmp]=x,++val[x];while(tmp^x);} 38 } 39 #undef y 40 int ans1[N],ans2[N],vis[N],ans; 41 #define y rdag[j].to 42 int dp1(int x){//dbg2(x,val[x]); 43 if(vis[x])return ans1[x]; 44 vis[x]=1; 45 for(register int j=rhd[x];j;j=rdag[j].nxt)MAX(ans1[x],dp1(y)+val[x]); 46 return ans1[x]; 47 } 48 #undef y 49 #define y dag[j].to 50 int dp2(int x){ 51 if(vis[x])return ans2[x]; 52 vis[x]=1; 53 for(register int j=dhd[x];j;j=dag[j].nxt)MAX(ans2[x],dp2(y)+val[x]); 54 return ans2[x]; 55 } 56 #undef y 57 int main(){//freopen("test.in","r",stdin);//freopen("test.ans","w",stdout); 58 read(n),read(m); 59 for(register int i=1,x,y;i<=m;++i)read(x),read(y),Addedge(x,y); 60 for(register int i=1;i<=n;++i)if(!dfn[i])tarjan(i); 61 for(register int t=1,u,v;t<=tot1;++t){ 62 u=frm[t],v=G[t].to; 63 if(bel[u]^bel[v])AddDAG(bel[u],bel[v]),AddrDAG(bel[v],bel[u]);//CAUTION!LOOP! 64 } 65 fill(ans1+1,ans1+n+1,-INF),fill(ans2+1,ans2+n+1,-INF);ans1[bel[1]]=ans2[bel[1]]=0; 66 for(register int i=1;i<=n;++i)if(val[i])dp1(i); 67 memset(vis,0,sizeof vis);ans=val[bel[1]]; 68 for(register int i=1;i<=n;++i)if(val[i])dp2(i); 69 for(register int t=1,u,v;t<=tot2;++t){ 70 u=rdag[t].to,v=dag[t].to; 71 MAX(ans,ans1[v]+ans2[u]+val[bel[1]]); 72 } 73 printf("%d\n",ans); 74 return 0; 75 }
问题总结:缩点记忆化dp要常检查两样:初始化及边界、缩点是否影响了什么(会不会出错)。