题目链接

(Luogu) https://www.luogu.org/problem/P3756

(BZOJ) http://lydsy.com/JudgeOnline/problem.php?id=4823

题解

有点神仙的最小割题。

考虑题目里的图形,如果我们用四种颜色对棋盘进行染色,奇数行依次染\(0,1,2,3,0,1,2,3...\), 偶数行依次染\(3,2,1,0,3,2,1,0...\)则条件可以转化为不能出现相连的\(4\)个颜色互不相同的块。

那么可以建一个四层的图,对于每条两侧都有关键点的特殊边,按照\(S\rightarrow 3\rightarrow 0\rightarrow 1\rightarrow 2\rightarrow T\)的顺序连边,其中\(S\rightarrow 3\)连\(3\)色点的点权,\(0\rightarrow 1\)连两个关键点权值的最小值,\(2\rightarrow T\)连\(2\)色点的点权。不出现相连的四个颜色互不相同的块等价于不存在从\(S\)到\(T\)的路径。

然后跑最小割即可。

因为是分层图,所以dinic跑得很快(复杂度应该是\(O(n\sqrt n)\)),可以通过此题。

代码

#include<bits/stdc++.h>
#define llong long long
using namespace std; const int INF = 1e9;
namespace NetFlow
{
const int N = 1e5+2;
const int M = 8e5;
struct Edge
{
int v,w,nxt,rev;
} e[(M<<1)+3];
int fe[N+3];
int te[N+3];
int dep[N+3];
int que[N+3];
int n,en,s,t;
void addedge(int u,int v,int w)
{
// printf("addedge %d %d %d\n",u,v,w);
en++; e[en].v = v; e[en].w = w;
e[en].nxt = fe[u]; fe[u] = en; e[en].rev = en+1;
en++; e[en].v = u; e[en].w = 0;
e[en].nxt = fe[v]; fe[v] = en; e[en].rev = en-1;
}
bool bfs()
{
for(int i=1; i<=n; i++) dep[i] = 0;
int head = 1,tail = 1; que[1] = s; dep[s] = 1;
while(head<=tail)
{
int u = que[head]; head++;
for(int i=fe[u]; i; i=e[i].nxt)
{
int v = e[i].v;
if(e[i].w>0 && dep[v]==0)
{
dep[v] = dep[u]+1;
if(v==t)return true;
tail++; que[tail] = v;
}
}
}
return false;
}
int dfs(int u,int cur)
{
if(u==t||cur==0) {return cur;}
int rst = cur;
for(int &i=te[u]; i; i=e[i].nxt)
{
int v = e[i].v;
if(e[i].w>0 && rst>0 && dep[v]==dep[u]+1)
{
int flow = dfs(v,min(rst,e[i].w));
if(flow>0)
{
e[i].w -= flow;
rst -= flow;
e[e[i].rev].w += flow;
if(rst==0) {return cur;}
}
}
}
if(rst==cur) {dep[u] = -2;}
return cur-rst;
}
int dinic(int _n,int _s,int _t)
{
n = _n,s = _s,t = _t;
int ret = 0;
while(bfs())
{
for(int i=1; i<=n; i++) te[i] = fe[i];
memcpy(te,fe,sizeof(int)*(n+1));
ret += dfs(s,INF);
}
return ret;
}
}
using NetFlow::addedge;
using NetFlow::dinic; const int N = 1e5;
struct Point
{
int x,y,w;
} a[N+3];
map<int,int> mp[N+3];
int id[N+3],clr[N+3];
int n,nx,ny; int getclr(int x,int y)
{
if(y&1) {return (x-1)&3;}
else {return 3-((x-1)&3);}
} int main()
{
scanf("%d%d%d",&nx,&ny,&n);
for(int i=1; i<=n; i++)
{
scanf("%d%d%d",&a[i].x,&a[i].y,&a[i].w);
mp[a[i].x][a[i].y] = i;
}
for(int i=1; i<=n; i++)
{
int x = a[i].x,y = a[i].y,clr = getclr(x,y);
if(((x+(y<<1))&3)==3 && mp[x+1].count(y))
{
int j = mp[x+1][y],w = min(a[i].w,a[j].w);
if(clr==0)
{
int k = mp[x-1][y]; if(k) {addedge(k+2,i+2,INF);}
k = mp[x][y-1]; if(k) {addedge(k+2,i+2,INF);}
k = mp[x][y+1]; if(k) {addedge(k+2,i+2,INF);}
k = mp[x+2][y]; if(k) {addedge(j+2,k+2,INF);}
k = mp[x+1][y+1]; if(k) {addedge(j+2,k+2,INF);}
k = mp[x+1][y-1]; if(k) {addedge(j+2,k+2,INF);}
addedge(i+2,j+2,w);
}
else if(clr==1)
{
int k = mp[x+2][y]; if(k) {addedge(k+2,j+2,INF);}
k = mp[x+1][y+1]; if(k) {addedge(k+2,j+2,INF);}
k = mp[x+1][y-1]; if(k) {addedge(k+2,j+2,INF);}
k = mp[x-1][y]; if(k) {addedge(i+2,k+2,INF);}
k = mp[x][y-1]; if(k) {addedge(i+2,k+2,INF);}
k = mp[x][y+1]; if(k) {addedge(i+2,k+2,INF);}
addedge(j+2,i+2,w);
}
}
else if(clr==3)
{
addedge(1,i+2,a[i].w);
}
else if(clr==2)
{
addedge(i+2,2,a[i].w);
}
}
int ans = dinic(n+2,1,2);
printf("%d\n",ans);
return 0;
}

BZOJ 4823 Luogu P3756 [CQOI2017]老C的方块 (网络流、最小割)的更多相关文章

  1. BZOJ4823 [Cqoi2017]老C的方块 【最小割】

    题目 老C是个程序员. 作为一个懒惰的程序员,老C经常在电脑上玩方块游戏消磨时间.游戏被限定在一个由小方格排成的R行C列网格上,如果两个小方格有公共的边,就称它们是相邻的,而且有些相邻的小方格之间的公 ...

  2. 【洛谷P3756】[CQOI2017]老C的方块(最小割)

    洛谷 题意: 给出一个网格图类似于这样: 现在给出一个\(n*m\)大小的网格,之后会给出一些点,若某些点相连形成了如下的几个图案,那么就是不好的. 现在可以删去一些点,但删除每个点都有一些代价,问最 ...

  3. 洛谷$P3756\ [CQOI2017]$老$C$的方块 网络流

    正解:网络流 解题报告: 传送门$QwQ$ 看到不能出现给定的讨厌的图形,简单来说就,特殊边两侧的方格不能同时再连方格. 所以如果出现,就相当于是四种方案?就分别炸四个格子. 然后冷静分析一波之后发现 ...

  4. BZOJ4823 CQOI2017老C的方块(最小割)

    如果将其转化为一个更一般的问题即二分图带权最小单边点覆盖(最小控制集)感觉是非常npc的.考虑原题给的一大堆东西究竟有什么奇怪的性质. 容易发现如果与特殊边相邻的两格子都放了方块,并且这两个格子都各有 ...

  5. BZOJ 4823 Luogu P3756 老C的方块 染色+最小割

    题面太长了请各位自行品尝—>老C的方块 分析: 我们要解决掉所有使人弃疗的组合,还要保证花费最小,容易想到最小割(当然你要是想费用流的话,我们就没办法定义流量了) 我们来分析一下那些令人弃疗的组 ...

  6. [bzoj4823][洛谷P3756][Cqoi2017]老C的方块

    Description 老 C 是个程序员. 作为一个懒惰的程序员,老 C 经常在电脑上玩方块游戏消磨时间.游戏被限定在一个由小方格排成的R行C列网格上 ,如果两个小方格有公共的边,就称它们是相邻的, ...

  7. BZOJ 2039 / Luogu P1791 [2009国家集训队]employ人员雇佣 (最小割)

    题面 BZOJ传送门 Luogu传送门 分析 考虑如何最小割建图,因为这仍然是二元关系,我们可以通过解方程来确定怎么建图,具体参考论文 <<浅析一类最小割问题 湖南师大附中 彭天翼> ...

  8. bzoj 4823 [Cqoi2017]老C的方块——网络流

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4823 一个不合法方案其实就是蓝线的两边格子一定选.剩下两部分四相邻格子里各选一个. 所以这个 ...

  9. BZOJ 4823 [Cqoi2017]老C的方块 ——网络流

    lrd的题解:http://www.cnblogs.com/liu-runda/p/6695139.html 我还是太菜了.以后遇到这种题目应该分析分析性质的. 网络流复杂度真是$O(玄学)$ #in ...

随机推荐

  1. Wannafly挑战赛24

    A. 石子游戏 Alice和Bob在玩游戏,他们面前有n堆石子,对于这些石子他们可以轮流进行一些操作,不能进行下去的人则输掉这局游戏.可以进行两种操作:1. 把石子数为奇数的一堆石子分为两堆正整数个石 ...

  2. C#中word文档转html

    var path = Request.Url.Host + ":" + Request.Url.Port + list[i].AnnexPath; //html保存路径 strin ...

  3. C# 添加log4net日志

    一.添加log4net的Nuget包 二.在Web.config或者App.config文件中添加log4net配置 代码: <log4net> <!-- OFF, FATAL, E ...

  4. JS基础_条件运算符

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  5. 带她来看Frozen 2

  6. 采购订单保存生成PO号后增强点。

    EXIT_SAPMM06E_013 这个增强可用于生成的PO后,调用外部接口把变更或生成的PO信息下发出去. 这里面的参数 I_EKKO 是新的抬头 I_EKKO_OLD 是更改前的抬头 XEKPO ...

  7. sqoop 安装

    Sqoop是一款开源的工具,主要用于在Hadoop(Hive)与传统的数据库(mysql.postgresql...)间进行数据的传递,可以将一个关系型数据库(例如 : MySQL ,Oracle , ...

  8. SmartBinding与kbmMW#2

    前言 在之前的文章中,我介绍了SmartBinding作为Delphi的一个新的易于使用和智能的绑定框架.介绍了包括绑定对象,列表,常规数据和可视控件,以及如何使用导航器,所有这些都用代码做了演示. ...

  9. php底层的运行机制和原理

    php设计的理念及特点 PHP 被设计为一种适用于web开发的动态脚本语言,同时具有解释型和弱类型,底层完全由C语言实现. 解释型即程序边运行边解释,一行一行运行. 弱类型即变量类型一开始并不是确定的 ...

  10. mysql 使用zip包进行安装以及服务启动后立即关闭问题

             本实例使用的mysql版本为 mysql-8.0.15-winx64 1.下载zip包 官网地址:https://dev.mysql.com/downloads/mysql/ 2.安 ...