POJ2125 Destroying The Graph
题目链接:ヾ(≧∇≦*)ゝ
大致题意:
给出一个有向图D=(V,E).对于每个点U,定义两种操作a(u),b(u)
操作a(u):删除点U的所有出边,即属于E,操作花费为Ca(u).
操作b(u):删除点U的所有入边,即属于E,操作花费为Cb(u).
求将原图的边集的边全部删除的最小代价,总操作数和具体操作
Solution:
第一问很简单,首先,对于每一个点,把它分成出点和入点。
把每个点的出点与S相连,入点与T相连。边容量分别为删除该点所有入边和出边的花费。
然后对于每条边 a -> b,就把a的出点与b的入点连一条容量为inf的边。
根据最大流=最小割,跑一遍dinic就能得到答案了。
对于第二、三问,我们分别统计a操作和b操作。
我们先对剩余网络进行bfs(),把能够扫到的点都标记为1,不能的标记为0。
对于一个点u,如果要使用a(u),那么显然,需要至少存在一个点v,满足u -> v &&
vis[u]vis[v]0。
而对于点u,如果要使用b(u),只需要满足vis[u]==1就行了。
为了防止重复输出,在定义一个apr数组记录每个数是否加入到答案中就行了。
详见代码
Code:
#include<queue>
#include<ctype.h>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define N 1001
#define M 20001
#define inf 1926081700
using namespace std;
int S,T,head[N];
int n,m,cnt=1;
int ru[N],cu[N];
int ans,vis[N],apr[N];
int t1,t2,fst[N],sec[N];
struct Edge{int nxt,to,val;}edge[M];
void ins(int x,int y,int z){
edge[++cnt].nxt=head[x];
edge[cnt].to=y;edge[cnt].val=z;
head[x]=cnt;
}
namespace Network_Flow{
queue<int> q;
int dep[N];
int bfs(){
memset(dep,0,sizeof(dep));
q.push(S);dep[S]=1;
while(!q.empty()){
int x=q.front();q.pop();
for(int i=head[x];i;i=edge[i].nxt){
int y=edge[i].to,v=edge[i].val;
if(!dep[y]&&v){
q.push(y);
dep[y]=dep[x]+1;
}
}
}
return dep[T];
}
int dfs(int x,int rest){
if(x==T||rest<=0) return rest;
int flow=0;
for(int i=head[x];i;i=edge[i].nxt){
int y=edge[i].to,v=edge[i].val;
if(dep[y]==dep[x]+1&&v){
int now=dfs(y,min(rest,v));
edge[i].val-=now;
edge[i^1].val+=now;
flow+=now;rest-=now;
if(!rest) break;
}
}
return flow;
}
int dinic(){
int maxflow=0;
while(bfs()) maxflow+=dfs(S,inf);
return maxflow;
}
}
void getspj(){
queue<int> s;
s.push(S);vis[S]=1;
while(!s.empty()){
int x=s.front();s.pop();
for(int i=head[x];i;i=edge[i].nxt)
if(!vis[edge[i].to]&&edge[i].val){
s.push(edge[i].to);
vis[edge[i].to]=1;
}
}
apr[S]=apr[T]=1;
}
int read(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)){if(ch=='-')f=-f;ch=getchar();}
while(isdigit(ch)){x=x*10+ch-48;ch=getchar();}
return x*f;
}
int main(){
n=read(),m=read();
S=n*2+1,T=S+1;
for(int i=1;i<=n;i++) ru[i]=read();
for(int i=1;i<=n;i++) cu[i]=read();
for(int i=1;i<=n;i++){
ins(S,i,cu[i]);ins(i,S,0);
ins(i+n,T,ru[i]);ins(T,i+n,0);
}
for(int x,y,i=1;i<=m;i++){
x=read(),y=read();
ins(x,n+y,inf);
ins(n+y,x,0);
}
using namespace Network_Flow;
printf("%d\n",dinic());getspj();
for(int i=1;i<=n;i++)
for(int j=head[i];j;j=edge[j].nxt){
int y=edge[j].to;
if(!vis[i]&&!vis[y]&&!apr[i]){
sec[++t2]=i;
ans++;apr[i]=1;
}
if(!apr[y]&&vis[y]){
fst[++t1]=y%n;
if(!fst[t1]) fst[t1]=n;
ans++;apr[y]=1;
}
}
printf("%d\n",ans);
for(int i=1;i<=t1;i++) printf("%d +\n",fst[i]);
for(int i=1;i<=t2;i++) printf("%d -\n",sec[i]);
return 0;
}
POJ2125 Destroying The Graph的更多相关文章
- POJ2125 Destroying The Graph (最小点权覆盖集)(网络流最小割)
Destroying The Graph Time Limit: 2000MS Memo ...
- POJ2125 Destroying The Graph(二分图最小点权覆盖集)
最小点权覆盖就是,对于有点权的有向图,选出权值和最少的点的集合覆盖所有的边. 解二分图最小点权覆盖集可以用最小割: vs-X-Y-vt这样连边,vs和X部点的连边容量为X部点的权值,Y部和vt连边容量 ...
- POJ2125 Destroying The Graph 二分图 + 最小点权覆盖 + 最小割
思路来源:http://blog.csdn.net/lenleaves/article/details/7873441 求最小点权覆盖,同样求一个最小割,但是要求出割去了那些边, 只要用最终的剩余网络 ...
- POJ 2125 Destroying the Graph 二分图最小点权覆盖
Destroying The Graph Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 8198 Accepted: 2 ...
- 【POJ】【2125】Destroying the Graph
网络流/二分图最小点权覆盖 果然还是应该先看下胡伯涛的论文…… orz proverbs 题意: N个点M条边的有向图,给出如下两种操作.删除点i的所有出边,代价是Ai.删除点j的所有入边,代价是Bj ...
- 图论(网络流,二分图最小点权覆盖):POJ 2125 Destroying The Graph
Destroying The Graph Description Alice and Bob play the following game. First, Alice draws some di ...
- POJ 2125 Destroying The Graph [最小割 打印方案]
Destroying The Graph Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 8311 Accepted: 2 ...
- poj 2125 Destroying The Graph (最小点权覆盖)
Destroying The Graph http://poj.org/problem?id=2125 Time Limit: 2000MS Memory Limit: 65536K ...
- AC日记——Destroying The Graph poj 2125
Destroying The Graph Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 8356 Accepted: 2 ...
随机推荐
- handsontable 拖动末尾列至前面列位置,被拖动列消失的问题
问题描述:将最后一列在往前面列位置进行拖动后,被拖动的最后列消失掉了. 解决办法:在handsontabel绑定中去设置data值,取消通过 loadData 绑定data $("#topF ...
- AbelSu区块链应用场景收集
1.基于智能合约的众筹 众筹项目的资金通常由一个中心化不可变更且开放的数据库来控制,这个数据库可以追踪所有出资人. 虽然如此,我们可以用一种去中心化的方式来实现,而且只要创建一个代币就可以追踪资金.一 ...
- Flutter - 自动引用pub.dartlang.org/packages上最新的packages
一般在pubspec.yaml里面引用pub.dartlang.org/packages的packages时,我们都是在包名称后面加上版本号的,谷歌默认也是这样写的. cupertino_icons: ...
- XAMPP、PHPstorm和PHPcharm和Windows环境下Python搭建+暴力破解
XAMPP的安装和使用 一.什么是XAMPP? XAMPP是最流行的PHP开发环境. XAMPP是完全免费且易于安装的Apache发行版,其中包含Apache.MariaDB.PHP和Perl. 类似 ...
- 【php增删改查实例】第八节 - 部门管理模块(编写PHP程序)
首先,在同级目录新建一个query.php文件: 接着,去刷新页面,打开F12,NetWork,看看当前的请求能不能走到对应的php文件? 这就说明datagrid确实能够访问到query.php 只 ...
- python中获取执行脚本路径方法
1.sys.path[0]:获取执行脚本目录绝对路径 #每次执行脚本时,python会将执行脚本目录加入PYTHONPATH环境变量中(sys.path获取) #!/usr/bin/python3 i ...
- 记录:tf.saved_model 模块的简单使用(TensorFlow 模型存储与恢复)
虽然说 TensorFlow 2.0 即将问世,但是有一些模块的内容却是不大变化的.其中就有 tf.saved_model 模块,主要用于模型的存储和恢复.为了防止学习记录文件丢失或者蠢笨的脑子直接遗 ...
- Redmine 安装指南
第一种方式 (一键安装): 准备工作: 1.最小化安装CentOS7 2.更新YUM源 3.更新系统关闭防火墙 yum -y update systemctl stop firewalld syste ...
- 利用顺序栈解决括号匹配问题(c++)-- 数据结构
题目: 7-1 括号匹配 (30 分) 给定一串字符,不超过100个字符,可能包括括号.数字.字母.标点符号.空格,编程检查这一串字符中的( ) ,[ ],{ }是否匹配. 输入格式: 输入在一行 ...
- Scrum Meeting NO.2
Scrum Meeting No.2 1.会议内容 今天,我们对已经确定的任务进行了分配,并针对界面设计方面的细节进行讨论. 由于这周其它课程任务繁重(编译+数据库).前端的任务主要分配给编程能力较好 ...