先缩点,然后拆点,其实是很经典的一种操作,把不好做的点拆成边,然后我一开始想的是网络流,答案当然是增广次数,

但可以发现跑网络流的话不同的跑法增广次数不一样,不太好找最小的。我们可以换一种神奇的思路,跑最大费用流,

这样根据费用流每次都在最长路上增广保证每次都跑了尽量多的点,根据贪心原理可知这样是正确的。

详见http://blog.csdn.net/iamzky/article/details/41846687;

(一开始我一直错,发现自己增广用的dinic,模板打惯了就直接打上了。。。。忘了dinic是可以一次dfs实现多次增广的,没法统计增广次数。。我好蠢啊)

#include<iostream>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<cstdio>
using namespace std;
const int maxn=,inf=1e9;
struct edg{
int nxt,to,f,c;
}e[maxn],g[maxn];
int res,last[maxn],H[maxn],t1=,t2=,cas,dis[maxn],q[maxn],head,tail;
void add1(int x,int y){
++t1;e[t1].nxt=last[x];last[x]=t1;e[t1].to=y;
}
void add2(int x,int y,int z,int zz){
++t2;g[t2].nxt=H[x];H[x]=t2;g[t2].to=y;g[t2].f=z;g[t2].c=zz;
++t2;g[t2].nxt=H[y];H[y]=t2;g[t2].to=x;g[t2].f=;g[t2].c=-zz;
}
int scc,low[maxn],dfn[maxn],bel[maxn],cnt,sta[maxn],top,in[maxn];
void tarjan(int x){
low[x]=dfn[x]=++cnt;sta[++top]=x;in[x]=;
for(int i=last[x];i;i=e[i].nxt){
int v=e[i].to;
if(!dfn[v]){
tarjan(v);low[x]=min(low[x],low[v]);
}
else{
if(in[v])low[x]=min(low[x],dfn[v]);
} }
//cout<<"orz"<<endl;
if(low[x]==dfn[x]){
int now=;scc++;
while(now!=x){
now=sta[top--];in[now]=;
bel[now]=scc;
}
}
}
struct dui{
int from,to;
}tmp[maxn];
int pe[maxn],pv[maxn],mp[][],ans,vis[maxn],N,n,m,a,b,x,y,S,T,A[maxn],B[maxn],a1[maxn],b1[maxn];
void solve(){
while(){
memset(dis,-,sizeof(dis));
head=tail=;q[++tail]=S;dis[S]=;vis[S]=;
while(head!=tail){
head=(head+)%maxn;
int u=q[head];
for(int i=H[u];i;i=g[i].nxt){
int v=g[i].to;
if(g[i].f&&dis[v]<dis[u]+g[i].c){
dis[v]=dis[u]+g[i].c;
pe[v]=i;pv[v]=u;
if(!vis[v]){
vis[v]=;tail=(tail+)%maxn;
q[tail]=v;
}
}
}
vis[u]=;
}
if(dis[T]<)break;
int mm=inf;
for(int u=T;u!=S;u=pv[u])
mm=min(mm,g[pe[u]].f);
for(int u=T;u!=S;u=pv[u])
g[pe[u]].f-=mm,g[pe[u]^].f+=mm;
res+=dis[T]*mm;
if(dis[T]>)
++ans;
}
}
int main(){
cin>>cas;
while(cas--){
memset(mp,,sizeof(mp));
memset(last,,sizeof(last));
memset(H,,sizeof(H));
memset(dfn,,sizeof(dfn));
cin>>n>>m>>a>>b;
t1=t2=;scc=cnt=;
for(int i=;i<=a;++i){scanf("%d",&A[i]);}
for(int i=;i<=b;++i)scanf("%d",&B[i]);
for(int i=;i<=m;++i){
scanf("%d%d",&tmp[i].from,&tmp[i].to);
add1(tmp[i].from,tmp[i].to);
}
for(int i=;i<=n;++i){
if(!dfn[i])tarjan(i);
}
S=;T=scc*+;
for(int i=;i<=n;++i){
for(int j=last[i];j;j=e[j].nxt){
int u=bel[i],v=bel[e[j].to];
if(mp[u][v]||u==v)continue;
add2(u+scc+,v+,inf,);
mp[u][v]=mp[v][u]=;
}
}
for(int i=;i<=a;++i){
add2(S,+bel[A[i]],inf,);
}
for(int i=;i<=b;++i){
add2(+bel[B[i]]+scc,T,inf,);
}
for(int i=;i<=scc;++i){
add2(+i,+scc+i,,);add2(+i,+scc+i,inf,);
}
res=ans=;solve();
if(res!=scc)puts("no solution");
else{printf("%d\n",ans);}
}
//system("pause");
return ;
}

bzoj2893(费用流)的更多相关文章

  1. hdu-5988 Coding Contest(费用流)

    题目链接: Coding Contest Time Limit: 2000/1000 MS (Java/Others)     Memory Limit: 65536/65536 K (Java/Ot ...

  2. POJ2195 Going Home[费用流|二分图最大权匹配]

    Going Home Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 22088   Accepted: 11155 Desc ...

  3. BZOJ3130: [Sdoi2013]费用流[最大流 实数二分]

    3130: [Sdoi2013]费用流 Time Limit: 10 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 960  Solved: 5 ...

  4. 洛谷 1004 dp或最大费用流

    思路: dp方法: 设dp[i][j][k][l]为两条没有交叉的路径分别走到(i,j)和(k,l)处最大价值. 则转移方程为 dp[i][j][k][l]=max(dp[i-1][j][k-1][l ...

  5. Codeforces 730I [费用流]

    /* 不要低头,不要放弃,不要气馁,不要慌张 题意: 给两行n个数,要求从第一行选取a个数,第二行选取b个数使得这些数加起来和最大. 限制条件是第一行选取了某个数的条件下,第二行不能选取对应位置的数. ...

  6. zkw费用流+当前弧优化

    zkw费用流+当前弧优化 var o,v:..] of boolean; f,s,d,dis:..] of longint; next,p,c,w:..] of longint; i,j,k,l,y, ...

  7. 【BZOJ-4213】贪吃蛇 有上下界的费用流

    4213: 贪吃蛇 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 58  Solved: 24[Submit][Status][Discuss] Desc ...

  8. 【BZOJ-3638&3272&3267&3502】k-Maximum Subsequence Sum 费用流构图 + 线段树手动增广

    3638: Cf172 k-Maximum Subsequence Sum Time Limit: 50 Sec  Memory Limit: 256 MBSubmit: 174  Solved: 9 ...

  9. [bzoj4514]数字配对[费用流]

    今年SDOI的题,看到他们在做,看到过了一百多个人,然后就被虐惨啦... 果然考试的时候还是打不了高端算法,调了...几天 默默地yy了一个费用流构图: 源连所有点,配对的点连啊,所有点连汇... 后 ...

随机推荐

  1. NuGet打包,IIS自动发布

    NuGet学习笔记(1)——初识NuGet及快速安装使用 https://kb.cnblogs.com/page/143190/ NuGet学习笔记(2)——使用图形化界面打包自己的类库 https: ...

  2. vue的通讯与传递props emit (简单的弹框组件)

    props父把信息传递给子组件 1父组件 <template> <div class="hello"> <div id="app-3&quo ...

  3. selenium无界面chromedriver

    chromeDriver下载地址:http://chromedriver.storage.googleapis.com/index.html 谷歌浏览器Chrome和驱动程序的对照表https://b ...

  4. InstallShield 2015 安装 在vs2015

     网上很少注册InstallShield 2015  的方法,而且很多以前版本的注册也很笼统,今天我就说说几个细节上的问题.相信大家看了会有帮助,有问题回帖,我会及时跟上, 先说说我遇到的问题 安装: ...

  5. istio-jaeger-python调用链配置

    虽然,istio ingress controller已经生成了jaeger 记录所需要的信息,但是多个分布式之间没法清晰记录相互之间的依赖关系.所以相关的项目还需要加入特殊配置. 如:python- ...

  6. php中static静态关键字的使用方法和应用场景

    php中除了常规类和方法的使用,访问控制之外,还有静态关键字static,静态变量可以是局部变量也可以是全局变量,当一个程序段执行完毕时,静态变量并没有消失,它依然存在于内存中,下次在定义时还是以前的 ...

  7. Data Dictionary 数据字典

    数据字典是一种通用的程序设计方法.可以认为,不论什么程序,都是为了处理一定的主体,这里的主体可能是人员.商品(超子).网页.接口.数据库表.甚至需求分析等等.当主体有很多的属性,每种属性有很多的取值, ...

  8. c++11 初始化列表 bind function 示例

    // 111111111111.cpp: 定义控制台应用程序的入口点. // #include "stdafx.h" #include <iostream> #incl ...

  9. Two Sum II - Input array is sorted LT167

    Given an array of integers that is already sorted in ascending order, find two numbers such that the ...

  10. Spring Boot学习笔记:ApplicationEvent和ApplicationEventListener事件监听

    采用事件监听的好处 以用户注册的业务逻辑为例,用户在填写完信息表单后,提交信息到后台,后台对用户信息进行处理,然后给用户返回处理结果的信息. 如上图所示,用户在注册时,后台需要处理一些系列流程,实际业 ...