题目:My Brute

Seaco是一个漂亮的妹子,喜欢玩一款名叫My Brute的游戏。情人节快到了,starvae和xingxing都想邀请妹子过节,但是妹子只能陪一个啊,于是两个人决定打一架,用男人的方式对决,来一场My Brute吧!

一开始两个人都有n(n<100)只宠物,每个宠物有生命值,伤害值,每次两个人各派出一只宠物,starvae可以任意确定宠物的出场顺序,xingxing不可以。

每局开始starvae先打,hp<=0的一方输。

starvvae的第i个宠物(因为可以换顺序,也就是可能不是第i个出场的)赢vi分,否则减vi分。如果结束时starvae的分数大于0,starvae就赢了。

现在请你求出如果能赢,输出最大分数,和出场顺序和原顺序的相似度(如果最大分数有多种可能的安排方法,选择相似度最大的方法)。不能赢就输出Oh, I lose my dear seaco!

题解:@g3&wy%*¥…sd…#%…

我没做出来。。。因为窝读错题了。。。哭T^T。。。查题解不小心就查到了答案。。。我的英语绝对没救了。。。虽然不知道读对也不知道能不能做出来。。。。。

很好想的是源点连每一个starvae的宠物,流量是1,权值为0,xingxing的每个宠物的连汇点,流量是1,权值为0,两个人的宠物两两相连,能赢权值是v[i],否则是-v[i]。

但是这里求得是最大值,以前也做过求最大值,方法是加边加-w,但这里不行,因为边本来就是有正有负。

所以每次spfa求最短路改为求最长路。记得初始化时是-inf。

还有一个问题是。。。如何保证尽可能原顺序。。。于是有一个比较巧妙的方法,就是把所有v[i]*100,然后把每一个顺序不变的边,权值+1,因为点的个数是100以内的,所以能保证结果正确。。

#include <cstring>
#include <vector>
#include <queue>
#include <cstdio>
using namespace std;
typedef long long ll; const int MAXV = ;
const int INF = <<; struct Edge { int to, cap, cost, rev; };
vector<Edge> G[MAXV];
int dist[MAXV], prv[MAXV], pre[MAXV], in[MAXV];
queue<int> que; void addedge(int from, int to, int cap, int cost) {
G[from].push_back((Edge){to, cap, cost, G[to].size()});
G[to].push_back((Edge){from, , -cost, G[from].size()-});
} int min_cost_max_flow(int s, int t) { //, int f) {
int res = ;
int f = ;
while () { //f > 0) {
for (int i = ; i <= t; ++i) dist[i] = -INF, in[i] = ;
dist[s] = ;
while (!que.empty()) que.pop();
in[s] = ;
que.push(s); while (!que.empty()) {
int u = que.front(); que.pop(); in[u] = ;
for (int i = ; i < G[u].size(); ++i) {
Edge &e = G[u][i];
if (e.cap > && dist[e.to] < dist[u] + e.cost) {
dist[e.to] = dist[u] + e.cost;
prv[e.to] = u;
pre[e.to] = i;
if (in[e.to] == ) {
in[e.to] = ;
que.push(e.to);
}
}
}
} if (dist[t] == -INF) break; //return -1; int d = INF; // d = f;
for (int v = t; v != s; v = prv[v]) {
d = min(d, G[prv[v]][pre[v]].cap);
}
f += d;
res += d * dist[t];
for (int v = t; v != s; v = prv[v]) {
Edge &e = G[prv[v]][pre[v]];
e.cap -= d;
G[v][e.rev].cap += d;
}
}
return res;
} const int N = ;
int v[N], h[N], p[N], a[N], b[N]; bool win(int h, int p, int a, int b) { // hp damage
int x = h/b; if (h % b != ) x++;
int y = p/a; if (p % a != ) y++;
return x >= y;
} int main()
{
int n;
while (~scanf("%d", &n) && n) {
for (int i = ; i < n; ++i) scanf("%d", &v[i]);
for (int i = ; i < n; ++i) scanf("%d", &h[i]);
for (int i = ; i < n; ++i) scanf("%d", &p[i]);
for (int i = ; i < n; ++i) scanf("%d", &a[i]);
for (int i = ; i < n; ++i) scanf("%d", &b[i]); int src = , sink = n+n+; for (int i = src; i <= sink; ++i) G[i].clear(); for (int i = ; i < n; ++i) {
for (int j = ; j < n; ++j) {
int tmp = win(h[i], p[j], a[i], b[j]) ? v[i] : -v[i];
tmp *= ; if (i == j) tmp++;
addedge(i+, n+j+, , tmp);
}
} for (int i = ; i < n; ++i) {
addedge(src, i+, , );
addedge(i++n, sink, , );
} int ans = min_cost_max_flow(src, sink); if (ans/ <= ) printf("Oh, I lose my dear seaco!\n");
else printf("%d %.3f%%\n", ans/, ans%/(double)n*); }
return ;
}

既然很明显是二分图

所以可以考虑一下KM解法,建图什么的都一样辣,速度还是一如既往的完爆费用流。。

#include <cstring>
#include <vector>
#include <queue>
#include <cstdio>
using namespace std;
typedef long long ll;
const int MAXN = ;
const int INF = 0x3f3f3f3f; int G[MAXN][MAXN];
int vx[MAXN], vy[MAXN];
bool visx[MAXN], visy[MAXN];
int match[MAXN];
int slack[MAXN]; int N; bool dfs(int x)
{
visx[x] = true; for (int y = ; y < N; ++y) { if (visy[y]) continue; int gap = vx[x] + vy[y] - G[x][y]; if (gap == ) {
visy[y] = true;
if (match[y] == - || dfs( match[y] )) {
match[y] = x;
return true;
}
} else {
slack[y] = min(slack[y], gap);
}
} return false;
} int KM()
{
memset(match, -, sizeof match);
memset(vy, , sizeof vy); for (int i = ; i < N; ++i) {
vx[i] = G[i][];
for (int j = ; j < N; ++j) {
vx[i] = max(vx[i], G[i][j]);
}
} for (int i = ; i < N; ++i) { fill(slack, slack + N, INF); while () {
memset(visx, false, sizeof visx);
memset(visy, false, sizeof visy); if (dfs(i)) break; int d = INF;
for (int j = ; j < N; ++j)
if (!visy[j]) d = min(d, slack[j]); for (int j = ; j < N; ++j) {
if (visx[j]) vx[j] -= d;
if (visy[j]) vy[j] += d;
else slack[j] -= d;
}
}
} int res = ;
for (int i = ; i < N; ++i)
res += G[ match[i] ][i]; return res;
} int v[MAXN], h[MAXN], p[MAXN], a[MAXN], b[MAXN]; bool win(int h, int p, int a, int b) { // hp damage
int x = h/b; if (h % b != ) x++;
int y = p/a; if (p % a != ) y++;
return x >= y;
} int main()
{
int n;
while (~scanf("%d", &n) && n) { N = n;
for (int i = ; i < n; ++i) scanf("%d", &v[i]);
for (int i = ; i < n; ++i) scanf("%d", &h[i]);
for (int i = ; i < n; ++i) scanf("%d", &p[i]);
for (int i = ; i < n; ++i) scanf("%d", &a[i]);
for (int i = ; i < n; ++i) scanf("%d", &b[i]); for (int i = ; i < n; ++i) {
for (int j = ; j < n; ++j) {
int tmp = win(h[i], p[j], a[i], b[j]) ? v[i] : -v[i];
tmp *= ; if (i == j) tmp++;
G[i][j] = tmp;
}
} int ans = KM(); if (ans/ <= ) printf("Oh, I lose my dear seaco!\n");
else printf("%d %.3f%%\n", ans/, ans%/(double)n*); }
return ;
}

hdu3315-My Brute(费用流 or KM算法)的更多相关文章

  1. poj2175费用流消圈算法

    题意:      有n个建筑,每个建筑有ai个人,有m个避难所,每个避难所的容量是bi,ai到bi的费用是|x1-x2|+|y1-y2|+1,然后给你一个n*m的矩阵,表示当前方案,问当前避难方案是否 ...

  2. POJ 2175:Evacuation Plan(费用流消圈算法)***

    http://poj.org/problem?id=2175 题意:有n个楼,m个防空洞,每个楼有一个坐标和一个人数B,每个防空洞有一个坐标和容纳量C,从楼到防空洞需要的时间是其曼哈顿距离+1,现在给 ...

  3. HDU 3315 My Brute(费用流)

    职务地址:HDU 3315 这个题的思路全然是自己想出来的,自我感觉挺巧妙的. . .(大牛勿喷.. . )对大胆建图又多了一份信心. 详细思路是构造一个二分图,Si连源点.Xi连汇点,流量都是1,费 ...

  4. 网络费用流-最小k路径覆盖

    多校联赛第一场(hdu4862) Jump Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Ot ...

  5. HDU 4862 Jump 费用流

    又是一个看了题解以后还坑了一天的题…… 结果最后发现是抄代码的时候少写了一个负号. 题意: 有一个n*m的网格,其中每个格子上都有0~9的数字.现在你可以玩K次游戏. 一次游戏是这样定义的: 你可以选 ...

  6. HDU 3488--Tour(KM or 费用流)

    因为每个点只能经过一次 所以考虑拆点 这题有坑,有重边.. KM算法 把一个点拆成入点和出点 入点在X部,出点在Y步. 如果u,v之间有路径,就在X部的u点连接Y部的v点 求完美匹配. 当完美匹配的时 ...

  7. 二分图带权匹配 KM算法与费用流模型建立

    [二分图带权匹配与最佳匹配] 什么是二分图的带权匹配?二分图的带权匹配就是求出一个匹配集合,使得集合中边的权值之和最大或最小.而二分图的最佳匹配则一定为完备匹配,在此基础上,才要求匹配的边权值之和最大 ...

  8. [BZOJ1937][SHOI2004]Mst最小生成树(KM算法,最大费用流)

    1937: [Shoi2004]Mst 最小生成树 Time Limit: 3 Sec  Memory Limit: 64 MBSubmit: 802  Solved: 344[Submit][Sta ...

  9. hdu 3395(KM算法||最小费用最大流(第二种超级巧妙))

    Special Fish Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Tota ...

随机推荐

  1. SaaS 公司如何应对 On-Call 挑战?

    Cloud Insight 集监控.管理.计算.协作.可视化于一身,帮助所有 IT 公司,减少在系统监控上的人力和时间成本投入,让运维工作更加高效.简单.本文系国内 ITOM 行业领军企业 OneAP ...

  2. PowerDesigner将name自动添加到Comment注释的方法 VB代码

    Option Explicit ValidationMode = True InteractiveMode = im_Batch Dim mdl ' the current model ' get t ...

  3. linux 模拟延时和丢包

    这是 RHCA 中的一个 BDP 的测试,这也是公司很常用的一种延时和丢包的模拟,现在分享给大家. 我们做的应用软件,还有测试 TCP/UDP  对比,测试 BDP 对 TCP/IP 的影响时,我们都 ...

  4. java中dao层和service层的区别是什么?

    首先解释面上意思,service是业务层,dao是数据访问层.呵呵,这个问题我曾经也有过,记得以前刚学编程的时候,都是在service里直接调用dao,service里面就new一个dao类对象,调用 ...

  5. Windows8、Windows8.1和Microsoft Office 2013激活

    KMS激活软件: 什么是KMS?什么是伪激活?http://tieba.baidu.com/p/3224677648 KMSpico激活软件:链接: http://pan.baidu.com/s/1e ...

  6. 理解extern char s[100]与extern char *s

    在x.c中定义了一个字符数组 char s[100],在l.c中进行引用extern char s[200], 有些c程序新手经常把它写成extern char *s. 这两种写法的含义一样吗? 首先 ...

  7. node.js 异步式I/O 与事件驱动

    Node.js 最大的特点就是异步式 I/O(或者非阻塞 I/O)与事件紧密结合的编程模式.这种模式与传统的同步式 I/O 线性的编程思路有很大的不同,因为控制流很大程度上要靠事件和回调函数来组织,一 ...

  8. 【HDOJ】4355 Party All the Time

    好久没做过三分的题目了. /* 4355 */ #include <iostream> #include <sstream> #include <string> # ...

  9. 关于pragma pack的用法(一)

    一个很重要的参数#pragma pack(n) 数据边界对齐方式:以如下结构为例: struct {                    char a;                    WOR ...

  10. cf A. Inna and Pink Pony(思维题)

    题目:http://codeforces.com/contest/374/problem/A 题意:求到达边界的最小步数.. 刚开始以为是 bfs,不过数据10^6太大了,肯定不是... 一个思维题, ...