题意:

在一个有向图中,有n个点,m条边$n \le 1000 \And \And  m \le 100000$

每条边有一个破坏的花费,有些点可以被选择并获得对应的金币。

假设一个可以选的点是$x$,你只有破坏了从1号点到$x$的路,才可以得到$x$对应的金币。

思路:

点和边的区别在于,你放弃一个点得到0,选择一个点得到val。你放弃选择一条边得到0,选择一条边得到 - val。

那么我们先把每个点的权值加到ans中,最够减去这个图的最小割即可。

由于要输出破坏边的方案,所以要dfs染色,又由于是有向边,所以破坏的边是起点属于s的边。

// #pragma GCC optimize(2)
// #pragma GCC optimize(3)
// #pragma GCC optimize(4)
#include <algorithm>
#include <iterator>
#include <iostream>
#include <cstring>
#include <cstdlib>
#include <iomanip>
#include <bitset>
#include <cctype>
#include <cstdio>
#include <string>
#include <vector>
#include <stack>
#include <cmath>
#include <queue>
#include <list>
#include <map>
#include <set>
#include <cassert>
//#include <unordered_set>
//#include <unordered_map>
// #include<bits/extc++.h>
// using namespace __gnu_pbds;
using namespace std;
#define pb push_back
#define fi first
#define se second
#define debug(x) cerr<<#x << " := " << x << endl;
#define bug cerr<<"-----------------------"<<endl;
#define FOR(a, b, c) for(int a = b; a <= c; ++ a) typedef long long ll;
typedef long double ld;
typedef pair<int, int> pii;
typedef pair<ll, ll> pll; const int inf = 0x3f3f3f3f;
const ll inff = 0x3f3f3f3f3f3f3f3f;
const int mod = 1e9+; template<typename T>
inline T read(T&x){
x=;int f=;char ch=getchar();
while (ch<''||ch>'') f|=(ch=='-'),ch=getchar();
while (ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
return x=f?-x:x;
} /**********showtime************/
const int maxn = ;
int head[maxn], gtot = ;
struct Edge{
int u, v, w;
int id;
int nxt;
} edge[];
void addedge(int u, int v, int w, int id) {
edge[gtot].u = u;
edge[gtot].v = v;
edge[gtot].w = w;
edge[gtot].nxt = head[u];
edge[gtot].id = id;
head[u] = gtot++; edge[gtot].u = v;
edge[gtot].v = u;
edge[gtot].w = ;
edge[gtot].nxt = head[v];
edge[gtot].id = ;
head[v] = gtot++;
}
int n,m,f;
vector<int>vec;
int cur[maxn],dis[maxn];
bool bfs(int s, int t) {
for(int i=s; i<=t; i++) {
cur[i] = head[i];
dis[i] = inf;
}
dis[s] = ; queue<int>que;
que.push(s);
while(!que.empty()) {
int u = que.front(); que.pop();
for(int i=head[u]; ~i; i=edge[i].nxt) {
int v = edge[i].v, w = edge[i].w;
if(w > && dis[v] > dis[u] + ) {
dis[v] = dis[u] + ;
que.push(v);
}
}
}
return dis[t] < inf;
}
int dfs(int u, int t, int maxflow) {
if(u == t || maxflow == ) return maxflow;
for(int i=cur[u]; ~i; i=edge[i].nxt) {
cur[u] = i;
int v = edge[i].v, w = edge[i].w;
if(w > && dis[v] == dis[u] + ) {
int f = dfs(v, t, min(maxflow, w));
if(f > ) {
edge[i].w -= f;
edge[i ^ ].w += f;
return f;
}
}
}
return ;
}
int dinic(int s, int t) {
int flow = ;
while(bfs(s, t)) {
while(int f = dfs(s, t, inf))
flow += f;
}
return flow;
}
int col[maxn];
void colnode(int u) {
col[u] = ;
for(int i=head[u]; ~i; i= edge[i].nxt) {
int v = edge[i].v, w = edge[i].w;
if(w && col[v] == ) colnode(v);
}
}
int main(){
int T; scanf("%d", &T);
int cas = ;
while(T--) {
scanf("%d%d%d", &n, &m, &f);
int s = , t = n + ;
for(int i=s; i<=t; i++) head[i] = -, col[i] = ;
gtot = ; for(int i=; i<=m; i++) {
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
addedge(u, v, w, i);
} int sum = ;
addedge(s, , inf, );
for(int i=; i<=f; i++) {
int u, w;
scanf("%d%d", &u, &w);
addedge(u, t, w, );
sum += w;
} sum = sum - dinic(s, t);
printf("Case %d: %d\n", ++cas, sum);
colnode(s);
vec.clear();
for(int i=; i<gtot; i++) {
int u = edge[i].u, v = edge[i].v, w = edge[i].w;
int id = edge[i].id;
if(col[u] == && col[v] == && id) {
vec.pb(id);
}
}
printf("%d", vec.size());
// sort(vec.begin(), vec.end());
for(int v : vec) {
printf(" %d", v);
}
puts("");
}
return ;
}

[HDU 3521] [最小割] Being a Hero的更多相关文章

  1. Being a Hero (hdu 3251 最小割 好题)

    Being a Hero Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) T ...

  2. hdu 4289(最小割)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4289 思路:求最小花费,最小割应用,将点权转化为边权,拆点,(i,i+n)之间连边,容量为在城市i的花 ...

  3. hdu 5076 最小割灵活运用

    这意味着更复杂的问题,关键的事实被抽象出来:每个点,能够赋予既有的值(挑两个一.需要选择,设定ai,bi). 寻找所有和最大.有条件:如果两个点同时满足: 1,:二进制只是有一个不同之处.  2:中的 ...

  4. Game HDU - 3657(最小割)

    Game Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

  5. hdu 1565 最小割

    黑白染色,源指向白,黑指向汇,容量都是方格中数的大小,相邻的格子白指向黑,容量为oo,然后求一次最小割. 这个割是一个简单割,如果只选择不在割中的点,那么一种割就和一个选数方案一一对应,割的大小就是不 ...

  6. hdu 3657 最小割的活用 / 奇偶方格取数类经典题 /最小割

    题意:方格取数,如果取了相邻的数,那么要付出一定代价.(代价为2*(X&Y))(开始用费用流,敲升级版3820,跪...) 建图:  对于相邻问题,经典方法:奇偶建立二分图.对于相邻两点连边2 ...

  7. hdu 3657 最小割(牛逼!!!!)总算理解了

    <strong></strong> 转载:http://blog.csdn.net/me4546/article/details/6662959 加颜色的太棒了!!! 在网上看 ...

  8. hdu 3691最小割将一个图分成两部分

    转载地址:http://blog.csdn.net/xdu_truth/article/details/8104721 题意:题给出一个无向图和一个源点,让你求从这个点出发到某个点最大流的最小值.由最 ...

  9. hdu 1569 最小割

    和HDU 1565是一道题,只是数据加强了,貌似轮廓线DP来不了了. #include <cstdio> #include <cstring> #include <que ...

随机推荐

  1. POJ 1182-食物链 [并查集] 《挑战程序设计竞赛》2.4

    Description 动物王国中有三类动物A,B,C,这三类动物的食物链构成了有趣的环形.A吃B, B吃C,C吃A. 现有N个动物,以1-N编号.每个动物都是A,B,C中的一种,但是我们并不知道它到 ...

  2. Spring Data JPA 介绍

    Spring-data-jpa的基本介绍:JPA诞生的缘由是为了整合第三方ORM框架,建立一种标准的方式,百度百科说是JDK为了实现ORM的天下归一,目前也是在按照这个方向发展,但是还没能完全实现.在 ...

  3. linux新建文件夹

    mkdir -p .... -p  ----parents no error if existion, make parent directories as needed

  4. jvm的学习笔记:二、类的初始化,代码实战(2)

    常量在编译阶段,会存在调用这个常量的方法的所在的类的常量池当中 System.out.println(MyParent2.str); 输出: hello parent2 依据:在MyTest2类调用M ...

  5. python 连接Oracle 的步骤

    1. 安装 cx_Oracle pip install cx_Oracle 2.配置  oci.dll 与 oraociei11.dll 添加到环境变量path中 下载地址:百度搜索下载,Oracle ...

  6. 【VS开发】获得devcon.exe

    1.获得devcon.exe 有两种方法,一是直接去网上下,不过下的很多64位的都不能用,二是自己装个ddk去安装目录下找,在WinDDK\7600.16385.1\tools\devcon下,当然还 ...

  7. C学习笔记-makefile

    除非最简单的项目,任何一个商业项目都会包含多个源代码,而且在编译的时候会有很长很复杂的指令,为了编译的简单可行,makefile由此而生,在编译时候直接输入make便会按照makefile里面的规则编 ...

  8. Redhat7 安装 yum(换成免费版) 安装gcc

    最近上Linux系统基础课程,要在虚拟机上编译运行程序,这时候就需要安装gcc,网上一搜,各种什么在线,离线安装,其中在线安装很方面,一个命令 yum install gcc 即可解决 可我这么输入后 ...

  9. Duilib的控件拖拽排序,支持跨容器拖拽(网易云信版本)

    完整代码见:https://github.com/netease-im/NIM_Duilib_Framework/pull/151 核心代码(思路): appitem.h #pragma once # ...

  10. __VA_ARGS__用法

    转载 自定义调试信息的输出 调试信息的输出方法有很多种,  例如直接用printf,  或者出错时使用perror, fprintf等将信息直接打印到终端上, 在Qt上面一般使用qDebug,而守护进 ...