hdu多校第二场1008(hdu6598) Harmonious Army 最小割
题意:
一个军队有n人,你可以给他们每个人安排战士或者法师的职业,有m对人有组合技,组合技的信息是a,b,c,代表如果这两个人是两个战士,则组合技威力为a,一个战士一个法师,威力为b,其中b=a/4+c/3,两个法师,威力为c,求最大的威力。
题解:
这道题给出了一个最小割的清奇思路。
首先不去想边权,只想连通性,设源点为战士,汇点为法师,中间的点记为军队中的人,人和源点汇点之间连边,有组合技的人两两之间连边,简单起见,从只有两个人的情况讨论,这张图割完了只能是如下形式:

(两个战士)

(两个法师)

(1战士2法师)

(1法师2战士)
我们假想,求出了最小割,说明了每个人只选一种职业后的情况。那么,被割掉的,应该是这两个人选择其他职业的威力之和。
那么,边权怎么算呢?
可列出如下方程组
(源,1)+(源,2)=b+c
(源,1)+(2,汇)+(1,2)=(1,汇)+(源,2)+(1,2)=a+c
(1,汇)+(2,汇)=a+b
得到一组解
(源,1)=(源,2)=(a+b)/2
(1,汇)=(2,汇)=(c+b)/2
(1,2)=(a-2b+c)/2
以此建边,重复的边就是权值之和,求出最小割后,用所有职业组合的威力之和减去最小割,就是答案。
注意,题目数据保证了上述值都为正,但是做类似题时思路时不要僵化,必须是上述方程组解为正时才可应用此法。
#include<bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
#define rep(i, a, b) for(int i=(a); i<(b); i++)
#define per(i, a, b) for(int i=(b)-1; i>=(a); i--)
#define sz(a) (int)a.size()
#define de(a) cout << #a << " = " << a << endl
#define dd(a) cout << #a << " = " << a << " "
#define all(a) a.begin(), a.end()
#define pw(x) (1ll<<(x))
#define lb(x) ((x) & -(x))
#define endl "\n"
typedef double db;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int, int> pii;
typedef vector<int> vi;
const int P = 1e9 + ;
int add(int a, int b) {if((a += b) >= P) a -= P; return a < ? a + P : a;}
int mul(int a, int b) {return 1ll * a * b % P;}
int kpow(int a, int b) {int r=;for(;b;b>>=,a=mul(a,a)) {if(b&)r=mul(r,a);}return r;}
//---- const int N = 1e3 + ;
const db eps = 1e-;
template<class T>
struct Dinic{
const static int N = , M = N * ;
int s , t , n , h[N] , cur[N] , lv[N] , q[N] , e , ne[M] , to[M];
T cap[M] , flow;
void liu(int u,int v,T w){ to[e] = v;ne[e] = h[u];cap[e] = w;h[u] = e++;}
void link(int u,int v,T w){ liu(u , v , w);liu(v , u , );}
void ini(int _n = N) { fill(h , h + (n=_n) , -);e = ;}
bool bfs(){
int L = , R = ;
fill(lv , lv + n , -);
lv[q[R++] = s] = ;
while(L < R && !~lv[t]){
int c = q[L++];
for(int k = h[c]; ~k ; k = ne[k])
if(cap[k] > && !~lv[to[k]])
lv[q[R++] = to[k]] = lv[c] + ;
}
return ~lv[t];
}
T dfs(int c,T mx){
if(c == t) return mx;
T ret = ;
for(int &k = cur[c]; ~k; k = ne[k]){
if(lv[to[k]] == lv[c] + && cap[k] > ){
T flow = dfs(to[k] , min(mx , cap[k]));
ret += flow;cap[k] -= flow , cap[k^] += flow;mx -= flow;
if (!mx) return ret;
}
}
lv[c] = -;
return ret;
}
T run(int _s,int _t){
s = _s , t = _t;
flow = ;
while(bfs()){
copy(h , h + n , cur);
flow += dfs(s, ~0U>>);
}
return flow;
}
}; Dinic<db> G; ll n, m, s, t, u, v, a, b, c, ans, w[N], ww[N]; int main() {
std::ios::sync_with_stdio(false);
std::cin.tie(); while (cin >> n >> m) {
ans=;
rep(i,,n+) w[i]=ww[i]=; G.ini(n + ); s = G.n - ; t = s + ;
rep(i, , m+) {
cin >> u >> v >> a >> b >> c;
w[u] += a + b; w[v] += a + b;
ww[u] += b + c; ww[v] += b + c;
G.link(u, v, - b + a / 2.0 + c / 2.0);
G.link(v, u, - b + a / 2.0 + c / 2.0);
if (u > v) swap(u, v);
ans += a + c + b;
}
rep(i, , n+) G.link(s, i, w[i] / 2.0), G.link(i, t, ww[i] / 2.0);
ans = (ll)round(ans - G.run(s, t));
cout << ans << endl;
} return ;
}
hdu多校第二场1008(hdu6598) Harmonious Army 最小割的更多相关文章
- 2019年杭电多校第二场 1008题Harmonious Army(HDU6598+最小割+建图)
题目链接 传送门 题意 有\(n\)个士兵,要你给他们分配职业.有\(m\)对关系,对于某一对关系\(u,v\),如果同为勇士则总能力增加\(a\),同法师则增加\(c\),一个勇士一个法师增加\(\ ...
- [2019杭电多校第二场][hdu6598]Harmonious Army(最小割)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6598 题意是说一个军队有n人,你可以给他们每个人安排战士或者法师的职业,有m对人有组合技,组合技的信息 ...
- [2019杭电多校第一场][hdu6582]Path(最短路&&最小割)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6582 题意:删掉边使得1到n的最短路改变,删掉边的代价为该边的边权.求最小代价. 比赛时一片浆糊,赛后 ...
- 2019HDU多校赛第二场 H HDU 6598 Harmonious Army(最小割模型)
参考博客https://blog.csdn.net/u013534123/article/details/97142191 #include<bits/stdc++.h> using na ...
- HDU6602 Longest Subarray hdu多校第二场 线段树
HDU6602 Longest Subarray 线段树 传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6602 题意: 给你一段区间,让你求最长的区间使 ...
- hdu多校第二场 1005 (hdu6595) Everything Is Generated In Equal Probability
题意: 给定一个N,随机从[1,N]里产生一个n,然后随机产生一个n个数的全排列,求出n的逆序数对的数量,加到cnt里,然后随机地取出这个全排列中的一个非连续子序列(注意这个子序列可以是原序列),再求 ...
- hdu多校第二场1009 (hdu6599) I Love Palindrome String 回文自动机/字符串hash
题意: 找出这样的回文子串的个数:它本身是一个回文串,它的前一半也是一个回文串 输出格式要求输出l个数字,分别代表长度为1~l的这样的回文串的个数 题解: (回文自动机和回文树是一个东西) 首先用回文 ...
- hdu多校第二场1011 (hdu6601) Keen On Everything But Triangle 主席树
题意: 给定一个数列,每次询问一个区间,问这个区间中的值可组成的周长最大的三角形的周长. 题解: 定理1:给定一些值,这些值中组成边长最大的三角形的三条边的大小排名一定是连续的. 证明:假如第k大,第 ...
- hdu多校第二场 1010 (hdu6600)Just Skip This Problem
题意: 给你一个数x,允许你多次询问yi,然后回答你x xor yi 是否等于yi,询问尽量少的次数以保证能求出xi是几,求出这样询问次数最少的询问方案数. 结果mod1e6+3 题解: 队友赛时很快 ...
随机推荐
- 【python+selenium自动化】图像识别技术在UI自动化测试中的实际运用
引言: 目前在图像识别方面的自动化测试框架有很多,其中比较有名的是airtest,主要做手机端的游戏自动化测试(http://airtest.netease.com/) 因为没有实际把airtest运 ...
- auto_ptr与shared_ptr
注: 从c++11开始, auto_ptr已经被标记为弃用, 常见的替代品为shared_ptr shared_ptr的不同之处在于引用计数, 在复制(或赋值)时不会像auto_ptr那样直接转移所有 ...
- C#中拼音模糊匹配汉字智能搜索
准备: 微软官方出了一个专用的汉字转拼音包Microsoft Visual Studio International Pack 1.0 SR1 首先到官网http://www.microsoft.co ...
- 在使用spring中的ContextConfiguration、test注解时出现的错误
错误: 在使用测试注解时出现ContextConfiguration注解和test注解无法正常导包使用的编译异常,如图: 解决办法: 将pom.xml文件中以下依赖管理 中的<scope> ...
- python之callable
callback是python的内置函数 英文说明: callable(object) Return True If the object argument appears callable,Fals ...
- WOJ#2423 安全出行Safe Travel
描述 精灵最近在农场上泛滥,它们经常会阻止牛们从农庄(牛棚_1)走到别的牛棚(牛_i的目的 地是牛棚_i).每一个精灵只认识牛_i并且知道牛_i一般走到牛棚_i的最短路经.所以它们在牛_i到牛棚_i之 ...
- [BZOJ1492] [NOI2007] 货币兑换Cash(cdq分治+斜率优化)
[BZOJ1492] [NOI2007] 货币兑换Cash(cdq分治+斜率优化) 题面 分析 dp方程推导 显然,必然存在一种最优的买卖方案满足:每次买进操作使用完所有的人民币:每次卖出操作卖出所有 ...
- Codeforces 191C (LCA+树上差分算法)
题面 传送门 题目大意: 给出一棵树,再给出k条树上的简单路径,求每条边被不同的路径覆盖了多少次 分析 解决这个问题的经典做法是树上差分算法 它的思想是把"区间"修改转化为左右端点 ...
- Codeforces - 1198D - Rectangle Painting 1 - dp
https://codeforces.com/contest/1198/problem/D 原来是dp的思路,而且是每次切成两半向下递归.好像在哪里见过类似的,貌似是紫书的样子. 再想想好像就很显然的 ...
- asp.net 关于SessionId
原文:https://www.cnblogs.com/zhang1999/p/7278020.html 登陆页面使用Session存储验证码,导致会话产生SessionId,从而导致会话固定,登陆后用 ...