有上下界的费用流

#include <stdio.h>
#include <algorithm>
#include <queue>
#include <cstring> // begin{最小费用最大流}
struct edge {int from, to, cap, flow, cost, next;}; const int MAXN = 50 + 7;
const int inf = 0x3f3f3f3f; edge e[MAXN * MAXN + 7];
int head[MAXN * 2 + 7];
int e_sz, supS, supT, sum_l; inline void add_edge(int from, int to, int cap, int cost) {
if(!cap) return;
// printf("%d %d %d %d\n", from, to, cap, cost);
e[e_sz].from = from, e[e_sz].to = to, e[e_sz].cap = cap, e[e_sz].cost = cost, e[e_sz].flow = 0, e[e_sz].next = head[from]; head[from] = e_sz++;
e[e_sz].from = to, e[e_sz].to = from, e[e_sz].cap = 0, e[e_sz].cost = -cost, e[e_sz].flow = 0, e[e_sz].next = head[to]; head[to] = e_sz++;
} inline void add_edge_with_s(int u, int v, int l, int h, int c) {
//printf("edge_with_strict %d %d %d %d %d\n", u, v, l, h, c);
sum_l += l; add_edge(supS, v, l, c); add_edge(u, supT, l, c); add_edge(u, v, h - l, c);
} std::queue<int>q;
int dis[MAXN * 2], pre[MAXN * 2]; bool vis[MAXN * 2];
bool spfa(int s, int t) {
for(int i = 0; i < MAXN * 2; ++i)dis[i] = inf, vis[i] = false, pre[i] = -1;
// printf("%d %d\n", s, t);
dis[s] = 0, vis[s] = true; q.push(s);
while(!q.empty()) {
int u = q.front(); q.pop(); vis[u] = false;
for(int i = head[u], v; i != -1; i = e[i].next) {
v = e[i].to;
if(e[i].cap > e[i].flow && dis[v] > dis[u] + e[i].cost) {
dis[v] = dis[u] + e[i].cost, pre[v] = i;
if(!vis[v]) vis[v] = true, q.push(v);
}
}
}
return (pre[t] != -1);
} //返回的是最大流, cost存的是最小费用
int minCostMaxflow(int s, int t, int &cost) {
int flow = 0, Min; cost = 0;
while(spfa(s, t)) {
Min = inf;
for(int i = pre[t]; i != -1; i = pre[e[i ^ 1].to])
if(Min > e[i].cap - e[i].flow) Min = e[i].cap - e[i].flow;
for(int i = pre[t]; i != -1; i = pre[e[i ^ 1].to])
e[i].flow += Min, e[i ^ 1].flow -= Min, cost += e[i].cost * Min;
flow += Min;
}
return flow;
}
// end{最小费用最大流} int mat[MAXN][MAXN], Rl[MAXN], Rh[MAXN], Cl[MAXN], Ch[MAXN], cntR[MAXN], cntC[MAXN];
int main() {
int n,s,t;
while(~scanf("%d", &n)) {
memset(head, -1, sizeof(head)); e_sz = 0;
memset(cntR, 0, sizeof(cntR)); memset(cntC, 0, sizeof(cntC));
s = n * 2 + 1, t = n * 2 + 2;
supS = n * 2 + 3, supT = n * 2 + 4, sum_l = 0;
for(int i = 0; i < n; ++i) for(int j = 0; j < n; ++j)scanf("%d", &mat[i][j]);
for(int i = 0; i < n; ++i) scanf("%d%d", Rl + i, Rh + i);
for(int i = 0; i < n; ++i) scanf("%d%d", Cl + i, Ch + i); for(int i = 0; i < n; ++i) {
for(int j = 0; j < n; j++) cntR[i] += mat[i][j], cntC[i] += mat[j][i];
add_edge_with_s(s, i + 1, cntR[i], cntR[i], 0), add_edge_with_s(s, i + 1 + n, cntC[i], cntC[i], 0);
//add_edge(s, i + 1, cntR[i], 0), add_edge(s, i + 1 + n, cntC[i], 0);
add_edge_with_s(i + 1, t, Rl[i], Rh[i], 0), add_edge_with_s(i + n + 1, t, Cl[i], Ch[i], 0);
}
add_edge(t, s, inf, 0);
for(int i = 0, x1, x2, y1, y2; i < n * n / 2; ++i) {
scanf("%d%d%d%d", &x1, &y1, &x2, &y2); x1--, y1--, x2--, y2--;
if(mat[x1][y1] == mat[x2][y2])continue;
if(!mat[x1][y1]) std::swap(x1, x2), std::swap(y1, y2);
if(x1 == x2) add_edge(y1 + 1 + n, y2 + 1 + n, 1, 1);
else if(y1 == y2) add_edge(x1 + 1, x2 + 1, 1, 1);
}
//printf("input done\n");
int cost, flow = minCostMaxflow(supS, supT, cost);
// printf("total flow = %d\n", flow);
if(flow != sum_l) puts("-1");
else printf("%d\n", cost);
}
return 0;
}

[HihoCoder-1424] Asa's Chess Problem的更多相关文章

  1. 【hihocoder 1424】 Asa's Chess Problem(有源汇上下界网络流)

    UVALive-7670 ICPC北京2016-C题 hihocoder 1424 题意 有个 \(N\times N\) 的棋盘,告诉你每个格子黑色(1)或白色(0),以及每对能相互交换的同行或同列 ...

  2. Asa's Chess Problem

    一.题目 给定一张 \(n\times n\) 的矩阵,每个点上面有黑棋或者是白棋,给定 \(\frac{n\times n}{2}\) 对可以交换的位置,每对位置一定在同一行 \(/\) 同一列.\ ...

  3. UVa 750 - 8 Queens Chess Problem

    题目大意:八皇后问题,在一个8*8的棋盘上,放置8个皇后,使得任意两个皇后不在同一行上.不在同一列上.不在同一条对角线上,不过这道题预先给定了一个位置放置一个皇后,让你输出所有可能的答案. 经典的回溯 ...

  4. [2019HDU多校第二场][HDU 6591][A. Another Chess Problem]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6591 题目大意:二维坐标系上,所有满足\(5|2x+y\)的点都被设为障碍物,无法通过.现给出一对点, ...

  5. The 2016 ACMICPC Asia Beijing Regional Contest

    A. Harmonic Matrix Counter (3/19) B. Binary Tree (1/14) C. Asa's Chess Problem (21/65) [ Problem ] 给 ...

  6. HDU 5742 Chess SG函数博弈

    Chess Problem Description   Alice and Bob are playing a special chess game on an n × 20 chessboard. ...

  7. 2016暑假多校联合---A Simple Chess

    2016暑假多校联合---A Simple Chess   Problem Description There is a n×m board, a chess want to go to the po ...

  8. dp - Codeforces Round #313 (Div. 1) C. Gerald and Giant Chess

    Gerald and Giant Chess Problem's Link: http://codeforces.com/contest/559/problem/C Mean: 一个n*m的网格,让你 ...

  9. HDU 4405:Aeroplane chess(概率DP入门)

    http://acm.split.hdu.edu.cn/showproblem.php?pid=4405 Aeroplane chess Problem Description   Hzz loves ...

随机推荐

  1. 2018CCPCFINAL B Balance of the Force 枚举最大值

    题意 n个人能选择黑暗面和光明面,选择两个面分别能获得\(L_i\)和\(R_i\)的力量,有m对人不能选择同一面,问n个人的力量中的最大值-最小值尽可能小为多少. \(1<=n<=2\t ...

  2. 集合家族——List集合汇总

    一.概述 List继承了Collection,是有序的列表. 可重复数据 实现类有ArrayList.LinkedList.Vector.Stack等 ArrayList是基于数组实现的,是一个数组队 ...

  3. uiautomatorviewer真机使用报错Error obtaining UI hierarchy

    Mac OS+Android真机 8.0在使用uiautomatorviewer获取界面时报Error obtaining UI hierarchy Reason: Error while obtai ...

  4. MySQL数据分析(16)— 数据操作之增删改查

    前面我们说学习MySQL要从三个层面,四大逻辑来学,三个层面就是库层面,表层面和数据层面对吧,数据库里放数据表,表里放数据是吧,大家可以回忆PPT中jacky的这图,我们已经学完了库层面和表层面,从本 ...

  5. vue 使用 echart ,自定义样式案例

    1.vue 安装 echart 库 npm install echarts --save 2.vue代码 引入 let echarts = require("echarts/lib/echa ...

  6. Python学习日记(七)——装饰器

    1.必备知识 #### 一 #### def foo(): print 'foo' foo #表示是函数 foo() #表示执行foo函数 #### 二 #### def foo(): print ' ...

  7. mysql —日志记录

    日志 事务日志: transaction log 中继日志: reley log错误日志: error log 通用日志: general log 慢查询日志: slow query log 二进制日 ...

  8. jQuery源码解读----part 2

    分离构造器 通过new操作符构建一个对象,一般经过四步: A.创建一个新对象 B.将构造函数的作用域赋给新对象(所以this就指向了这个新对象) C.执行构造函数中的代码 D.返回这个新对象 最后一点 ...

  9. pycharm2019没有database问题(关于社区版)

    原文链接:https://blog.csdn.net/BlacK_CaT_/article/details/53884806网上教程都是直接打开右上角的database,但是我死活也没找到,后来发现应 ...

  10. v-on绑定特性命名带小横杠 ‘-’与props属性中变量怎么对应

    特性命名问题: 矛盾点一:html的特性不区分大小写 矛盾点二:Vue中除了模板命名,其他命名不允许出现小横杠 ‘-’ 在js文件内,命名为驼峰式,camerCase,进入html文件,自动转换成短横 ...