题目链接

LOJ

洛谷

容易想到最小费用最大流分配度数。

因为水管形态固定,每个点还是要拆成4个点,分别当前格子表示向上右下左方向。

然后能比较容易地得到每种状态向其它状态转移的费用(比如原向上的可以流到向下)。

注意比如向左向上的L,左连右,上连下,没有上连右(日常zz)。

可以看这的图

解决旋转的问题后,还要处理流量从哪里产生、结束。

因为是网格图,容易想到黑白染色。题目中"没有漏水水管"即格子的断头两两匹配,而匹配只发生在黑白格之间。so源点向所有白格子连边,所有黑格子向汇点连边。

因为匹配关系是确定的,所以即使相邻不一定有水管相连,匹配边还是要连的。

SPFA单路增广好慢啊,学一波多路增广。

可以,很快。

Update:我好像刚知道多路增广就是zkw费用流。。

朴素SPFA:

//7048kb	11328ms
#include <queue>
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
//#define gc() getchar()
#define MAXIN 200000
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
#define OK(i,j) (1<=(i)&&(i)<=n&&1<=(j)&&(j)<=m)
const int N=1e4+5,M=N*30; int n,m,src,des,Enum,H[N],nxt[M],fr[M],to[M],cap[M],cost[M],pre[N];
std::queue<int> q;
char IN[MAXIN],*SS=IN,*TT=IN; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
inline void AE(int u,int v,int c,bool flag)
{
if(flag) std::swap(u,v);//黑→白 把边反向
to[++Enum]=v, fr[Enum]=u, nxt[Enum]=H[u], cost[Enum]=c, cap[Enum]=1, H[u]=Enum;
to[++Enum]=u, fr[Enum]=v, nxt[Enum]=H[v], cost[Enum]=-c, cap[Enum]=0, H[v]=Enum;
}
bool SPFA()
{
static int dis[N];
static bool inq[N];
memset(dis,0x3f,sizeof dis);
dis[src]=0, q.push(src);
while(!q.empty())
{
int x=q.front();
q.pop(), inq[x]=0;//!...
for(int v,i=H[x]; i; i=nxt[i])
if(cap[i] && dis[v=to[i]]>dis[x]+cost[i])
pre[v]=i, dis[v]=dis[x]+cost[i], !inq[v]&&(q.push(v),inq[v]=1);
}
return dis[des]<0x3f3f3f3f;
}
inline int Augment()
{
int res=0;
for(int i=des; i!=src; i=fr[pre[i]])
res+=cost[pre[i]], --cap[pre[i]], ++cap[pre[i]^1];
return res;
}
int MCMF(int &cost)
{
int res=0;
while(SPFA()) cost+=Augment(), ++res;
return res;
} int main()
{
n=read(),m=read(); int tot=0;
int id[n+1][m+1][4];
for(int i=1; i<=n; ++i)
for(int j=1; j<=m; ++j)
for(int k=0; k<4; ++k) id[i][j][k]=++tot;
Enum=1, src=0, des=++tot;
bool f; int flow=0;
for(int i=1; i<=n; ++i)
for(int j=1,s; j<=m; ++j)//0上 1右 2下 3左
{//左 下 右 上
s=read(), f=(i+j)&1;
int u=f?des:src,up=id[i][j][0],ri=id[i][j][1],down=id[i][j][2],le=id[i][j][3];
if(s&1) AE(u,up,0,f), flow+=f^1;
if(s&2) AE(u,ri,0,f), flow+=f^1;
if(s&4) AE(u,down,0,f), flow+=f^1;
if(s&8) AE(u,le,0,f), flow+=f^1;
// if(!f)
// for(int k=0; k<4; ++k)
// if(s>>k&1) AE(src,id[i][j][k],0,0), ++flow;
// else ;//else!
// else for(int k=0; k<4; ++k) if(s>>k&1) AE(id[i][j][k],des,0,0);
if(!f)
{
if(OK(i-1,j)) AE(up,id[i-1][j][2],0,0);
if(OK(i,j-1)) AE(le,id[i][j-1][1],0,0);
if(OK(i+1,j)) AE(down,id[i+1][j][0],0,0);
if(OK(i,j+1)) AE(ri,id[i][j+1][3],0,0);
}
switch(s)
{
case 0: break;
case 1: AE(up,le,1,f), AE(up,ri,1,f), AE(up,down,2,f); break;
case 2: AE(ri,up,1,f), AE(ri,down,1,f), AE(ri,le,2,f); break;
case 3: AE(up,down,1,f), AE(ri,le,1,f); break;
case 4: AE(down,le,1,f), AE(down,ri,1,f), AE(down,up,2,f); break;
case 5: break;
case 6: AE(ri,le,1,f), AE(down,up,1,f); break;
case 7: AE(up,le,1,f), AE(down,le,1,f), AE(ri,le,2,f); break;
case 8: AE(le,up,1,f), AE(le,down,1,f), AE(le,ri,2,f); break;
case 9: AE(le,ri,1,f), AE(up,down,1,f); break;
case 10: break;
case 11: AE(le,down,1,f), AE(ri,down,1,f), AE(up,down,2,f); break;
case 12: AE(le,ri,1,f), AE(down,up,1,f); break;
case 13: AE(up,ri,1,f), AE(down,ri,1,f), AE(le,ri,2,f); break;
case 14: AE(le,up,1,f), AE(ri,up,1,f), AE(down,up,2,f); break;
case 15: break;
}
}
int cost=0;
if(MCMF(cost)==flow) printf("%d\n",cost);
else puts("-1"); return 0;
}

多路增广:

//5872kb	184ms
#include <queue>
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
//#define gc() getchar()
#define MAXIN 200000
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
#define OK(i,j) (1<=(i)&&(i)<=n&&1<=(j)&&(j)<=m)
const int N=1e4+5,M=N*30; int n,m,src,des,Enum,H[N],cur[N],nxt[M],to[M],cap[M],cost[M],dis[N],Cost;
std::queue<int> q;
bool vis[N];
char IN[MAXIN],*SS=IN,*TT=IN; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
inline void AE(int u,int v,int c,bool flag)
{
if(flag) std::swap(u,v);//黑→白 把边反向
to[++Enum]=v, nxt[Enum]=H[u], cost[Enum]=c, cap[Enum]=1, H[u]=Enum;
to[++Enum]=u, nxt[Enum]=H[v], cost[Enum]=-c, cap[Enum]=0, H[v]=Enum;
}
bool SPFA()
{
memset(vis,0,sizeof vis);
memset(dis,0x3f,sizeof dis);
dis[src]=0, q.push(src);
while(!q.empty())
{
int x=q.front();
q.pop(), vis[x]=0;
for(int v,i=H[x]; i; i=nxt[i])
if(cap[i] && dis[v=to[i]]>dis[x]+cost[i])
dis[v]=dis[x]+cost[i], !vis[v]&&(q.push(v),vis[v]=1);
}
return dis[des]<0x3f3f3f3f;
}
int DFS(int x/*int f*/)
{
if(x==des) return 1;
vis[x]=1;
for(int &i=cur[x]; i; i=nxt[i])
if(!vis[to[i]] && cap[i] && dis[to[i]]==dis[x]+cost[i])
if(DFS(to[i]))
return --cap[i], ++cap[i^1], Cost+=cost[i], 1;
return 0;
}
int MCMF()
{
int flow=0;
while(SPFA())
{
for(int i=src; i<=des; ++i) cur[i]=H[i];
while(DFS(src)) ++flow;
}
return flow;
} int main()
{
n=read(),m=read(); int tot=0;
int id[n+1][m+1][4];
for(int i=1; i<=n; ++i)
for(int j=1; j<=m; ++j)
for(int k=0; k<4; ++k) id[i][j][k]=++tot;
Enum=1, src=0, des=++tot;
bool f; int flow=0;
for(int i=1; i<=n; ++i)
for(int j=1,s; j<=m; ++j)//0上 1右 2下 3左
{//左 下 右 上
s=read(), f=(i+j)&1;
int u=f?des:src,up=id[i][j][0],ri=id[i][j][1],down=id[i][j][2],le=id[i][j][3];
if(s&1) AE(u,up,0,f), flow+=f^1;
if(s&2) AE(u,ri,0,f), flow+=f^1;
if(s&4) AE(u,down,0,f), flow+=f^1;
if(s&8) AE(u,le,0,f), flow+=f^1;
// if(!f)
// for(int k=0; k<4; ++k)
// if(s>>k&1) AE(src,id[i][j][k],0,0), ++flow;
// else ;//else!
// else for(int k=0; k<4; ++k) if(s>>k&1) AE(id[i][j][k],des,0,0);
if(!f)
{
if(OK(i-1,j)) AE(up,id[i-1][j][2],0,0);
if(OK(i,j-1)) AE(le,id[i][j-1][1],0,0);
if(OK(i+1,j)) AE(down,id[i+1][j][0],0,0);
if(OK(i,j+1)) AE(ri,id[i][j+1][3],0,0);
}
switch(s)
{
case 0: break;
case 1: AE(up,le,1,f), AE(up,ri,1,f), AE(up,down,2,f); break;
case 2: AE(ri,up,1,f), AE(ri,down,1,f), AE(ri,le,2,f); break;
case 3: AE(up,down,1,f), AE(ri,le,1,f); break;
case 4: AE(down,le,1,f), AE(down,ri,1,f), AE(down,up,2,f); break;
case 5: break;
case 6: AE(ri,le,1,f), AE(down,up,1,f); break;
case 7: AE(up,le,1,f), AE(down,le,1,f), AE(ri,le,2,f); break;
case 8: AE(le,up,1,f), AE(le,down,1,f), AE(le,ri,2,f); break;
case 9: AE(le,ri,1,f), AE(up,down,1,f); break;
case 10: break;
case 11: AE(le,down,1,f), AE(ri,down,1,f), AE(up,down,2,f); break;
case 12: AE(le,ri,1,f), AE(down,up,1,f); break;
case 13: AE(up,ri,1,f), AE(down,ri,1,f), AE(le,ri,2,f); break;
case 14: AE(le,up,1,f), AE(ri,up,1,f), AE(down,up,2,f); break;
case 15: break;
}
}
if(MCMF()==flow) printf("%d\n",Cost);
else puts("-1"); return 0;
}

BZOJ.5120.[清华集训2017]无限之环(费用流zkw 黑白染色)的更多相关文章

  1. LOJ 2321 清华集训2017 无限之环 拆点+最小费用最大流

    题面:中文题面,这里不占用篇幅 分析: 看到题面,我就想弃疗…… 但是作为任务题单,还是抄了题解…… 大概就是将每个格子拆点,拆成五个点,上下左右的触点和一个负责连源汇点的点(以下简称本点). 这个这 ...

  2. [清华集训2017]无限之环(infinityloop)

    description 题面 solution 一开始的思路是插头\(DP\),然而复杂度太高 考虑将网格图黑白染色后跑费用流 流量为接口数,费用为操作次数 把一个方格拆成五个点,如何连边请自行脑补 ...

  3. BZOJ 5120: [2017国家集训队测试]无限之环(费用流)

    传送门 解题思路 神仙题.调了一个晚上+半个上午..这道咋看咋都不像图论的题竟然用费用流做,将行+列为奇数的点和偶数的点分开,也就是匹配问题,然后把一个点复制四份,分别代表这个点的上下左右接头,如果有 ...

  4. BZOJ5120 [2017国家集训队测试]无限之环 费用流

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ5120 题意概括 原题挺简略的. 题解 本题好难. 听了任轩笛大佬<国家队神犇>的讲课才 ...

  5. Loj #2321. 「清华集训 2017」无限之环

    Loj #2321. 「清华集训 2017」无限之环 曾经有一款流行的游戏,叫做 *Infinity Loop***,先来简单的介绍一下这个游戏: 游戏在一个 \(n \times m\) 的网格状棋 ...

  6. [LOJ#2330]「清华集训 2017」榕树之心

    [LOJ#2330]「清华集训 2017」榕树之心 试题描述 深秋.冷风吹散了最后一丝夏日的暑气,也吹落了榕树脚下灌木丛的叶子.相识数年的Evan和Lyra再次回到了小时候见面的茂盛榕树之下.小溪依旧 ...

  7. [LOJ#2329]「清华集训 2017」我的生命已如风中残烛

    [LOJ#2329]「清华集训 2017」我的生命已如风中残烛 试题描述 九条可怜是一个贪玩的女孩子. 这天她在一堵墙钉了 \(n\) 个钉子,第 \(i\) 个钉子的坐标是 \((x_i,y_i)\ ...

  8. Loj #2331. 「清华集训 2017」某位歌姬的故事

    Loj #2331. 「清华集训 2017」某位歌姬的故事 IA 是一名会唱歌的女孩子. IOI2018 就要来了,IA 决定给参赛选手们写一首歌,以表达美好的祝愿.这首歌一共有 \(n\) 个音符, ...

  9. Loj #2324. 「清华集训 2017」小 Y 和二叉树

    Loj #2324. 「清华集训 2017」小 Y 和二叉树 小Y是一个心灵手巧的OIer,她有许多二叉树模型. 小Y的二叉树模型中,每个结点都具有一个编号,小Y把她最喜欢的一个二叉树模型挂在了墙上, ...

随机推荐

  1. ubuntu 下 teamview 取消自动启动 autostart

    sudo teamviewer daemon disable

  2. java多线程系列五、并发容器

    一.ConcurrentHashMap 1.为什么要使用ConcurrentHashMap 在多线程环境下,使用HashMap进行put操作会引起死循环,导致CPU利用率接近100%,HashMap在 ...

  3. Oracle把逗号分割的字符串转换为可放入in的条件语句的字符数列

    Oracle把逗号分割的字符串转换为可放入in的条件语句的字符数列   前台传来的字符串:'589,321' SELECT*FROM TAB_A T1 WHERE  T1.CODE  IN ( SEL ...

  4. redis安全 (error) NOAUTH Authentication required

    Redis 安全 我们可以通过 redis 的配置文件设置密码参数,这样客户端连接到 redis 服务就需要密码验证,这样可以让你的 redis 服务更安全. 实例 我们可以通过以下命令查看是否设置了 ...

  5. 《TCP/IP 详解 卷1:协议》第 3 章:链路层

    在体系结构中,我们知道:链路层(或数据链路层)包含为共享相同介质的邻居建立连接的协议和方法,同时,设计链路层的目的是为 IP 模块发送和接受 IP 数据报,链路层可用于携带支持 IP 的辅助性协议,例 ...

  6. 非root用户执行java进程报错:fork: retry:资源暂时不可用

    vim /etc/security/limits.conf # End of file *           soft   nproc        65535 *           hard   ...

  7. zookeeper3.4.6配置实现自动清理日志

    在使用zookeeper过程中,我们知道,会有dataDir和dataLogDir两个目录,分别用于snapshot和事务日志的输出(默认情况下只有dataDir目录,snapshot和事务日志都保存 ...

  8. 转载:磁盘目录(1.3.3)《深入理解Nginx》(陶辉)

    原文:https://book.2cto.com/201304/19614.html 要使用Nginx,还需要在Linux文件系统上准备以下目录. (1)Nginx源代码存放目录 该目录用于放置从官网 ...

  9. Linux服务器性能评估

    一.影响Linux服务器性能的因素 1. 操作系统级 CPU 内存 磁盘I/O带宽 网络I/O带宽 2. 程序应用级 二.系统性能评估标准 影响性能因素 影响性能因素 评判标准 好 坏 糟糕 CPU ...

  10. 蝉知CMS本地迁移到服务器具体步骤

    蝉知迁移步骤(2个方案,二选一即可) 方案一(整个chanzhi(eps)目录拷贝,假设新安装的蝉知文件夹名称为chanzhieps): 1.在新服务器上安装相同版本(版本号必须一致)的蝉知(安装文档 ...