P4177 [CEOI2008]order

斯德哥尔摩

2018-08-02 20:46:39

Personal

[P4177 [CEOI2008]order](https://www.luogu.org/problemnew/show/P4177) 首先如果没有租用这种鬼畜的操作,那么就是一个裸的最大权闭合子图例。 但是有了租用这种操作显然就那样来解决这个问题了 普通的最大权闭合子图将工作与源相连,流量为收益,机器与汇相连,流量为代价的相反数,对应的工作与机器之间是无穷大。 这样割掉与源点或汇点相连的边表示选择失去那种利益。 ~~感性认识一下,~~租用机器这种操作仅仅只会对一种工作产生影响,无论其他工作是否需要它。 这启发我们是不是可以用中间工作与机器的相连的边搞一些事情。。。 我们将工作与机器之间的边容量由正无穷变为租用的代价即可处理租用问题。 原理就是最大权闭合子图问题的实质是割掉一些边使源和汇不连通。 考虑源和汇之间的一条路径,有三种选择,对应割掉三条边。 显然个点左右两条边可以影响其他的路径,而割掉中间不会 因此我们就可以将中间的边转化为租用操作 其他的就和普通的最大权闭合子图一毛一样了。 附代码: ```cpp #include<iostream> #include<algorithm> #include<cstdio> #include<queue> #define MAXN 2410 #define MAX 999999999 using namespace std; int n,m,s,t,c=2,sum=0; int head[MAXN],deep[MAXN]; struct node{ int next,to,w; }a[MAXN*MAXN]; inline int read(){ int date=0,w=1;char c=0; while(c<'0'||c>'9'){if(c=='-')w=-1;c=getchar();} while(c>='0'&&c<='9'){date=date*10+c-'0';c=getchar();} return date*w; } inline void add(int u,int v,int w){ a[c].to=v;a[c].w=w;a[c].next=head[u];head[u]=c++; a[c].to=u;a[c].w=0;a[c].next=head[v];head[v]=c++; } bool bfs(){ int u,v; queue<int> q; for(int i=s;i<=t;i++)deep[i]=0; deep[s]=1; q.push(s); while(!q.empty()){ u=q.front(); q.pop(); for(int i=head[u];i;i=a[i].next){ v=a[i].to; if(a[i].w&&!deep[v]){ deep[v]=deep[u]+1; if(v==t)return true; q.push(v); } } } return false; } int dfs(int x,int limit){ if(x==t)return limit; int v,sum,cost=0; for(int i=head[x];i;i=a[i].next){ v=a[i].to; if(a[i].w&&deep[v]==deep[x]+1){ sum=dfs(v,min(a[i].w,limit-cost)); if(sum>0){ a[i].w-=sum; a[i^1].w+=sum; cost+=sum; if(limit==cost)break; } else deep[v]=-1; } } return cost; } int dinic(){ int ans=0; while(bfs())ans+=dfs(s,MAX); return ans; } void work(){ int ans=dinic(); printf("%d\n",sum-ans); } void init(){ int u,v,w; n=read();m=read(); s=0;t=n+m+1; for(int i=1;i<=n;i++){ w=read(); add(s,i,w); sum+=w; int x=read(); while(x--){ u=read();w=read(); add(i,u+n,w); } } for(int i=1;i<=m;i++){ w=read(); add(i+n,t,w); } } int main(){ init(); work(); return 0; } ```