首页 > 代码库 > bzoj3876: [Ahoi2014]支线剧情

bzoj3876: [Ahoi2014]支线剧情

神犇题解:http://blog.csdn.net/popoqqq/article/details/43024221

题意:给定一个DAG,1为起始点,任意一个点可以直接回到1,每条边有经过代价,求一种最优方案使得每条边至少经过一次,代价最小。

无源汇有下界最小费用可行流。

考虑我们对于无源汇的建图:对于每个点,它的入度下界为容量,S向它拉一条边,出度下界为容量,其向T连一条边,跑一边最大流即可。

这里我们建图实际上一样,但是加上了费用,具体建图如下:

对于x->y,费用为z

S向y建容量为1,费用为z的边(对应无源汇里的入度处理)

x向y建容量INF,费用为z的边(即自由流)

对于每一个点x,假设其出度为a

x向T建容量为a,费用为0的边(出度处理)(注意这里不要加上费用,因为已经在入度处理的时候就加过了)

x向1建容量INF,费用为0的边(对应题意)

然后跑最小费用流就可以了。

 

膜拜ihopenot大爷。

实际上有一个很大的建图优化,也是无源汇建图的优化,即合并入出度并且相互抵消。但是对于这道题就不好做,因为有费用,不过可以这么想:既然每一条边都要走,我们就直接把费用抽出来,即答案一开始就为所有边走一次的代价,然后这样入度的费用就变为0了。然后再与出度进行抵消,最后边的数量大大减少。

实践证明,从原来改到现在,时间从8000多优化到100多ms。

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 #define N 50005
 4 #define INF 1e9
 5 inline int read(){
 6     int x=0,f=1;char a=getchar();
 7     while(a<0 || a>9) {if(a==-) f=-1; a=getchar();}
 8     while(a>=0 && a<=9) x=x*10+a-0,a=getchar();
 9     return x*f;
10 }
11 int ans,n,cnt,a[N],p[N],d[N],head[N],S,T;
12 bool vis[N];
13 queue<int>q;
14 struct edges{
15     int fr,to,cap,flow,cost,next;
16 }e[2*N];
17 inline void insert(int u,int v,int f,int c){
18     e[cnt]=(edges){u,v,f,0,c,head[u]};head[u]=cnt++;
19     e[cnt]=(edges){v,u,0,0,-c,head[v]};head[v]=cnt++;
20 }
21 inline bool spfa(){
22     memset(d,0x3f,sizeof(d));
23     d[S]=0; a[S]=INF; q.push(S);
24     while(!q.empty()){
25         int x=q.front(); q.pop(); vis[x]=0;
26         for(int i=head[x];i>=0;i=e[i].next)
27             if(d[e[i].to]>d[x]+e[i].cost && e[i].flow<e[i].cap){
28                 d[e[i].to]=d[x]+e[i].cost; p[e[i].to]=i;
29                 a[e[i].to]=min(a[x],e[i].cap-e[i].flow);
30                 if(!vis[e[i].to]) vis[e[i].to]=1,q.push(e[i].to);
31             }
32     }
33     return d[T]<INF;
34 }
35 inline void mincf(){
36     ans+=a[T]*d[T];
37     int u=T;
38     while(u!=S){
39         e[p[u]].flow+=a[T];
40         e[p[u]^1].flow-=a[T];
41         u=e[p[u]].fr;
42     }
43 }
44 int main(){
45     memset(head,-1,sizeof(head));
46     n=read(); S=0; T=n+1;
47     for(int a,i=1;i<=n;i++){
48         a=read(); insert(i,T,a,0); 
49         if(i!=1) insert(i,1,INF,0);
50         for(int to,co,j=1;j<=a;j++)
51         to=read(),co=read(),insert(i,to,INF,co),insert(S,to,1,co);
52     }
53     while(spfa()) mincf();
54     printf("%d\n",ans);
55     return 0;
56 }

 

 

优化建图:

1 for(int a,i=1;i<=n;i++){
2         a=read(); cd[i]=a;
3         if(i!=1) insert(i,1,INF,0);
4         for(int to,co,j=1;j<=a;j++)
5         to=read(),co=read(),ans+=co,r[to]++,insert(i,to,INF,co);
6     }
7     for(int i=2;i<=n;i++) 
8     if(r[i]>cd[i]) insert(S,i,r[i]-cd[i],0);
9     else insert(i,T,cd[i]-r[i],0);

 

bzoj3876: [Ahoi2014]支线剧情