BZOJ5120 无限之环(费用流)
方案合法相当于要求接口之间配对,黑白染色一波,考虑网络流。有一个很奇怪的限制是不能旋转直线型水管,考虑非直线型水管有什么特殊性,可以发现其接口都是连续的。那么对于旋转水管,可以看做是把顺/逆时针方向上最后的接口提到最前。于是用四个点表示某格子的四个方向,以上述方式只移动一次就能相互转换的方向之间连费用1的边。然后在相邻格子可以相互匹配的方向之间连边,跑费用流即可。注意费用流的无向边必须按有向边的方式建反向边。不明白讨论一大串的是在干啥。
然后就非常悲惨了。多路增广的费用流会跑的很快,然而并不会。
- #include<iostream>
- #include<cstdio>
- #include<cmath>
- #include<cstdlib>
- #include<cstring>
- #include<algorithm>
- using namespace std;
- #define ll long long
- #define N 10010
- #define K 2010
- #define S 10001
- #define T 10002
- char getc(){char c=getchar();while ((c<'A'||c>'Z')&&(c<'a'||c>'z')&&(c<''||c>'')) c=getchar();return c;}
- int gcd(int n,int m){return m==?n:gcd(m,n%m);}
- int read()
- {
- int x=,f=;char c=getchar();
- while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
- while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
- return x*f;
- }
- int n,m,id[K][],p[N],d[N],q[N],pre[N],t=-,cnt,tot,ans,val;
- bool flag[N];
- struct data{int to,nxt,cap,flow,cost;
- }edge[N<<];
- void addedge(int x,int y,int z,int cost)
- {
- t++;edge[t].to=y,edge[t].nxt=p[x],edge[t].cap=z,edge[t].flow=,edge[t].cost=cost,p[x]=t;
- t++;edge[t].to=x,edge[t].nxt=p[y],edge[t].cap=,edge[t].flow=,edge[t].cost=-cost,p[y]=t;
- }
- inline int trans(int x,int y){return (x-)*m+y;}
- inline bool isblack(int x,int y){return (x&)^(y&);}
- inline int inc(int &x){x++;if (x>cnt+) x-=cnt+;return x;}
- bool spfa()
- {
- memset(d,,sizeof(d));d[S]=;
- memset(flag,,sizeof(flag));
- int head=,tail=;q[]=S;
- do
- {
- int x=q[inc(head)];flag[x]=;
- for (int i=p[x];~i;i=edge[i].nxt)
- if (d[x]+edge[i].cost<d[edge[i].to]&&edge[i].flow<edge[i].cap)
- {
- d[edge[i].to]=d[x]+edge[i].cost;
- pre[edge[i].to]=i;
- if (!flag[edge[i].to]) q[inc(tail)]=edge[i].to,flag[edge[i].to]=;
- }
- }while (head!=tail);
- return d[T]<N;
- }
- void ekspfa()
- {
- while (spfa())
- {
- int v=;
- for (int i=T;i!=S;i=edge[pre[i]^].to)
- if (edge[pre[i]].cap==edge[pre[i]].flow) {v=;break;}
- if (v)
- {
- ans++;
- for (int i=T;i!=S;i=edge[pre[i]^].to)
- val+=edge[pre[i]].cost,edge[pre[i]].flow++,edge[pre[i]^].flow--;
- }
- }
- }
- int main()
- {
- #ifndef ONLINE_JUDGE
- freopen("bzoj5120.in","r",stdin);
- freopen("bzoj5120.out","w",stdout);
- const char LL[]="%I64d\n";
- #else
- const char LL[]="%lld\n";
- #endif
- n=read(),m=read();
- memset(p,,sizeof(p));
- for (int i=;i<=n;i++)
- for (int j=;j<=m;j++)
- {
- int x=read(),u=,v=trans(i,j);
- for (int k=;k<;k++)
- {
- id[v][k]=++cnt;
- if (x&(<<k)) u++,isblack(i,j)?addedge(S,cnt,,):addedge(cnt,T,,);
- }
- tot+=u;
- if (x==||x==) continue;
- if (u==||u==) for (int k=;k<;k++) addedge(id[v][k],id[v][k+&],,),addedge(id[v][k+&],id[v][k],,);
- if (u==)
- {
- addedge(id[v][],id[v][],,),addedge(id[v][],id[v][],,);
- addedge(id[v][],id[v][],,),addedge(id[v][],id[v][],,);
- }
- }
- if (tot&) {cout<<-;return ;}tot>>=;
- for (int i=;i<=n;i++)
- for (int j=;j<=m;j++)
- if (isblack(i,j))
- {
- int v=trans(i,j);
- if (i>) addedge(id[v][],id[trans(i-,j)][],,);
- if (j<m) addedge(id[v][],id[trans(i,j+)][],,);
- if (i<n) addedge(id[v][],id[trans(i+,j)][],,);
- if (j>) addedge(id[v][],id[trans(i,j-)][],,);
- }
- ekspfa();
- if (ans<tot) cout<<-;
- else cout<<val;
- return ;
- }
BZOJ5120 无限之环(费用流)的更多相关文章
- BZOJ5120 [2017国家集训队测试]无限之环 费用流
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ5120 题意概括 原题挺简略的. 题解 本题好难. 听了任轩笛大佬<国家队神犇>的讲课才 ...
- BZOJ.5120.[清华集训2017]无限之环(费用流zkw 黑白染色)
题目链接 LOJ 洛谷 容易想到最小费用最大流分配度数. 因为水管形态固定,每个点还是要拆成4个点,分别当前格子表示向上右下左方向. 然后能比较容易地得到每种状态向其它状态转移的费用(比如原向上的可以 ...
- BZOJ 5120: [2017国家集训队测试]无限之环(费用流)
传送门 解题思路 神仙题.调了一个晚上+半个上午..这道咋看咋都不像图论的题竟然用费用流做,将行+列为奇数的点和偶数的点分开,也就是匹配问题,然后把一个点复制四份,分别代表这个点的上下左右接头,如果有 ...
- [BZOJ5120]无限之环
Description 曾经有一款流行的游戏,叫做InfinityLoop,先来简单的介绍一下这个游戏: 游戏在一个n×m的网格状棋盘上进行,其中有些小方格中会有水管,水管可能在方格某些方向的边界的中 ...
- 洛谷P4003 无限之环(infinityloop)(网络流,费用流)
洛谷题目传送门 题目 题目描述 曾经有一款流行的游戏,叫做 Infinity Loop,先来简单的介绍一下这个游戏: 游戏在一个 n ∗ m 的网格状棋盘上进行,其中有些小方格中会有水管,水管可能在格 ...
- LOJ2321. 「清华集训 2017」无限之环【费用流】
LINK 很好的一道网络里题 首先想插头DP的还是出门左转10分代码吧 然后考虑怎么网络流 首先要保证没有漏水 也就是说每个接口一定要有对应的接口 那么发现每个点只有可能和上下左右四个点产生联通关系 ...
- bzoj 5120 无限之环 & 洛谷 P4003 —— 费用流(多路增广SPFA)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=5120 https://www.luogu.org/problemnew/show/P4003 ...
- 洛谷P4003 无限之环(费用流)
传送门 神仙题啊……不看题解我可能一年都不一定做得出来……FlashHu大佬太强啦 到底是得有怎样的脑回路才能一眼看去就是费用流啊…… 建好图之后套个板子就好了,那么我们着重来讨论一下怎么建图 首先, ...
- LOJ 2321 清华集训2017 无限之环 拆点+最小费用最大流
题面:中文题面,这里不占用篇幅 分析: 看到题面,我就想弃疗…… 但是作为任务题单,还是抄了题解…… 大概就是将每个格子拆点,拆成五个点,上下左右的触点和一个负责连源汇点的点(以下简称本点). 这个这 ...
随机推荐
- ARP 地址分类 NAT技术
第1章 OSI回顾 1.1 TCP/IP协议族组成 应用层 主机到主机层 互联网层 网络接入层 1.2 总结应用层掌握的协议与端口号对应关系 http(80) telnet(23) ftp(2 ...
- python 实现redis订阅发布功能
redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对更多,包括string(字符串).list(链表).set(集合).zset(sorted set ...
- java学习(五)Number类、Math类
Number类 顾名思义嘛,搞数字的,以前也用到过,就是相当于内置了一堆数字的类嘛,用哪种类型的就引用下这个包就好了呗 Integer.Long.Byte.Double.Float.Short都是Nu ...
- Kubernetes服务发现之Service详解
一.引子 Kubernetes Pod 是有生命周期的,它们可以被创建,也可以被销毁,然后一旦被销毁生命就永远结束.通过ReplicationController 能够动态地创建和销毁Pod(列如,需 ...
- 有序链表转换二叉搜索树(LeetCode)
将一个按照升序排列的有序数组,转换为一棵高度平衡二叉搜索树. 本题中,一个高度平衡二叉树是指一个二叉树每个节点 的左右两个子树的高度差的绝对值不超过 1. 示例: 给定有序数组: [-10,-3,0, ...
- Java的POI的封装与应用
Java对Excel表格的导出一直是对我有种可怕噩梦的东西,每次对要建立行与列,并一个一个放值,我是从心底拒绝的. 处于项目需求,需要导出表格,于是找到网上一版很好的开发, <不想用POI?几行 ...
- Ajax中post后台.net MVC中Controller的路径问题。
今天使用ajax的post方法,去访问mvc的controller,url的传值是 "//ProductCatalog/PostShareInfo",发现无论如何都访问不到. 我实 ...
- beat冲刺(4/7)
目录 摘要 团队部分 个人部分 摘要 队名:小白吃 组长博客:hjj 作业博客:beta冲刺(4/7) 团队部分 后敬甲(组长) 过去两天完成了哪些任务 整理博客 ppt模板 接下来的计划 做好机动. ...
- 1001. A+B Format (20)的解题思路以及多源代码文件的尝试编写
前言 这几天刚学了多源代码文件的编译,因为想尝试使用一下这种方法,所以想用此编写这次作业的程序.正好可以learning by doing,在做当中学习新知识.(编译器为Dev-C++) github ...
- 饭来了小组Alpha冲刺阶段记录
一.第一天 日期:2018/6/13 1.1今日完成任务情况以及遇到的问题 侯晓东: 1.完成任务项:具体了解了微信小程序的开发流程,然后大致规划了我们项目的进度和完成节点:汇总组员的进度,写博客:画 ...