@description@

n 个士兵,每个士兵可以选择加入 A 组或 B 组。

有 m 个组合技可以增加整个军队的力量:第 i 个组合技涉及到士兵 ui 与 vi,当两人同时加入 A 组力量值增加 ai,同时加入 B 组力量值增加 ci,否则力量值增加 bi。

求每种安排士兵的方案中军队的力量值最大可以为多少。

Input

多组数据。

每组数据开头包含两个正整数 n(n≤500) 和 m(m≤10^4)。

接下来 m 行,每行包含 5 个整数 u,v,a,b,c(1≤u,v≤n,u≠v,1≤a,b,c≤4×10^6),含义如上。保证同一对 (u, v) 只会出现一次。

保证 n 总和不超过 5×10^3,m 总和不超过 5×10^4。

Output

对于每组数据,输出最大力量值。

Sample Input

3 2

1 2 8 3 3

2 3 4 3 6

Sample Output

12

@solution@

这道题这么多年过去。。。怕不是已经成了套路题。。。

【注:因为我懒所以本来下面应该有讲解的图,但是被我咕了】

我们考虑使用最小割,用所有收益的总和 - 最小割代表的最小代价。

考虑建图:s 向 x 连边,x 向 t 连边。如果割前一条表示选择 A 组,割后一条表示选择 B 组。

考虑 m 对关系,我们先设 (s, u) 容量为 e,(s, v) 容量为 f;(u, t) 容量为 g,(v, t) 容量为 h;(u, v) 容量为 p,(v, u) 容量为 q。

则接下来我们分类讨论 u, v 的选择情况,求出其应该舍弃的收益(即代价)与应该割的边的对应关系。

u 选 A 组,v 选 A 组:对应割 (s, u), (s, v);其代价为 b + c;其对应的边容量和为 e + f。

u 选 B 组,v 选 B 组:对应割 (u, t), (v, t);其代价为 a + b;其对应的边容量和为 g + h。

u 选 A 组,v 选 B 组:对应割 (s, u), (v, t), (v, u);其代价为 a + c;其对应的边容量和为 e + h + q。

u 选 B 组,v 选 A 组:对应割 (u, t), (s, v), (u, v);其代价为 a + c;其对应的边容量和为 g + f + p。

由上,可以得到 e + f = b + c 等方程。注意到只有 a, b, c 为常量,所以这个方程可能有很多解。

在此处,我们仅考虑一组特殊解,即令 p = q, e = f, g = h。此时可以解出 e = f = (b + c) / 2, g = h = (a + b) / 2, p = q = (a + c) / 2 - b。

于是,我们可以对于每个 x 统计与 x 有关的组合技中 (b + c) / 2 之和与 (a + b) / 2 之和(分别记为 s1[x], s2[x]),然后 s 向 x 连容量为 s1[x] 的边,x 向 t 连容量为 s2[x] 的边。

两个点之间如果有组合技,则连一条双向的、容量为 (a + c) / 2 - b 的边。

这样跑最小割就可以跑出最小代价。除以 2 什么的可以把所有容量乘 2 再最后除回来。

@accepted code@

#include<cstdio>
#include<algorithm>
using namespace std;
typedef long long ll;
const ll INF = (1LL<<60);
const int MAXN = 500;
const int MAXM = 10000;
const int MAXV = MAXN;
const int MAXE = 4*(MAXN + MAXM);
struct FlowGraph{
struct edge{
int to; ll cap, flow;
edge *nxt, *rev;
}edges[MAXE + 5], *adj[MAXV + 5], *ecnt;
int d[MAXV + 5], vd[MAXV + 5], s, t;
void init(int n) {
for(int i=1;i<=n;i++)
d[i] = vd[i] = 0, adj[i] = NULL;
ecnt = &edges[0];
}
void addedge(int u, int v, ll c) {
edge *p = (++ecnt), *q = (++ecnt);
p->to = v, p->cap = c, p->flow = 0;
p->nxt = adj[u], adj[u] = p;
q->to = u, q->cap = 0, q->flow = 0;
q->nxt = adj[v], adj[v] = q;
p->rev = q, q->rev = p;
}
ll aug(int x, ll tot) {
if( x == t ) return tot;
int mind = t + 1; ll sum = 0;
for(edge *p=adj[x];p;p=p->nxt) {
if( p->cap > p->flow ) {
if( d[p->to] + 1 == d[x] ) {
ll del = aug(p->to, min(tot - sum, p->cap - p->flow));
sum += del, p->flow += del, p->rev->flow -= del;
if( sum == tot || d[s] == t + 1 ) return sum;
}
mind = min(mind, d[p->to]);
}
}
if( sum == 0 ) {
vd[d[x]]--;
if( vd[d[x]] == 0 ) {
d[s] = t + 1;
return sum;
}
d[x] = mind + 1;
vd[d[x]]++;
}
return sum;
}
ll max_flow(int _s, int _t) {
s = _s, t = _t;
ll flow = 0;
while( d[s] != t + 1 )
flow += aug(s, INF);
return flow;
}
}G;
ll x[MAXN + 5], y[MAXN + 5];
int main() {
int n, m;
while( scanf("%d%d", &n, &m) == 2 ) {
G.init(n + 2);
for(int i=1;i<=n;i++)
x[i] = y[i] = 0;
ll ans = 0;
int s = n + 1, t = n + 2;
for(int i=1;i<=m;i++) {
int u, v, a, b, c;
scanf("%d%d%d%d%d", &u, &v, &a, &b, &c);
x[u] += a + b, x[v] += a + b;
y[u] += c + b, y[v] += c + b;
G.addedge(u, v, a + c - 2*b);
G.addedge(v, u, a + c - 2*b);
ans += a + b + c;
}
for(int i=1;i<=n;i++)
G.addedge(s, i, x[i]), G.addedge(i, t, y[i]);
printf("%lld\n", ans - G.max_flow(s, t)/2);
}
}

@details@

不要问我为什么即使用了 long long 还是跑得非常快,问就是 O(能过)。

原题目还给了 a, b, c 之间种种不可描述的关系。。。但其实也是用来迷惑人的。。。

最新文章

  1. opts=opts | |{}
  2. [ACM训练] 算法初级 之 基本算法 之 枚举(POJ 1753+2965)
  3. 如何查看文件是dos格式还是unix格式的?
  4. Mybatis 操作数据库的主键自增长
  5. java enum(枚举)使用详解 + 总结
  6. 解决tomcat一闪而过问题
  7. Linux查看系统性能命令
  8. validatebox实现多重规则验证
  9. 【ADO.NET】6、SQLHelper简单封装
  10. iBatis入手案例
  11. Ubuntu下nginx+uwsgi+flask的执行环境搭建
  12. 排序算法学习,python实现
  13. Linux(以CentOS6.5示例)下安装Oracle官方最新版JDK(JDK1.8)
  14. [洛谷P1196][NOI2002]银河英雄传说 - 带偏移量的并查集(1)
  15. Java基础知识拾遗(三)
  16. QT 应用程序测试
  17. idea2017.3最新破解方法
  18. iOS网络请求安全认证(JWT,RSA)
  19. 《纪念碑谷》(Monument Valley) 系列游戏的空间结构是如何设计的?
  20. 【转】为什么 MQTT 是最适合物联网的网络协议

热门文章

  1. Java review-basic1
  2. ucore os 初始化
  3. WIN7快捷键大全
  4. Oracle存储函数,存储过程
  5. CenOS SSH无密码登录
  6. xmlns详解(转载)
  7. 安装docker报错问题
  8. sqlserver 取月初月末的时间
  9. PHP学习(语言结构语句)
  10. SQLServer → 09:索引