[LOJ#3022][网络流]「CQOI2017」老 C 的方块
定义有特殊边相邻的格子颜色为黑,否则为白
可以看出,题目给出的限制条件的本质是如果两个小方块所在的格子 \(x\) 和 \(y\) 为两个相邻的黑格,那么 \(x\) 和 \(y\) 之间必然有一者满足其上下左右的所有白格内都没有小方块
即对于这样的 \(x\) 和 \(y\) ,需要以下三个条件至少满足一个:
(1)用 \(\min(w_x,w_y)\) 的代价删除 \(x\) 和 \(y\) 上的其中一个小方块
(2)把与 \(x\) 相邻的白格内的小方块全部删掉
(3)把与 \(y\) 相邻的白格内的小方块全部删掉
故考虑对于所有的白格 \(i\) ,拆成两个点 \(i_0\) 和 \(i_1\) ,并连边 \(<i_0,i_1,w_i>\)
对于所有的黑格组(相邻且之间有特殊边的两个黑格)\((x,y)\) ,也建立两个点,第一个点向第二个点连容量为 \(\min(w_x,w_y)\) 的边
这时候我们就有一个比较明显的最小割模型了
即对于需要「至少满足一个」的条件集合进行串联,需要「全部满足」的条件集合进行并联
具体地,如下图:

其中 \(x_1,x_2,x_3\) 表示与 \(x\) 相邻的白格, \(y_1,y_2,y_3\) 同理
以下设上图中 \(x_1,x_2,x_3\) 对应的三条边在第一层, \(\min(w_x,w_y)\) 对应的边在第二层, \(y_1,y_2,y_3\) 对应的三条边在第三层
至于 \(\min(w_x,w_y)\) 的边为什么一定要放在中间,后面会说
我们会发现一个问题:一个白格会属于 \(3\) 个黑格,直接建图会导致不同的黑格组之间互相影响
所以正确的建图方式是:设 \(x\) 在 \(y\) 左边,则要判断黑哥所在行号的奇偶性
若在奇行则 \(S\) 到 \(T\) 的路径上依次限制与 \(x\) 相邻的白格 \(\rightarrow\) 黑格组 \((x,y)\) 本身 \(\rightarrow\) 与 \(y\) 相邻的白格,否则反过来
考虑这样做的正确性:易得这么建图可以使得对于任意一个白格对应的限制边只在一层(第一层或第三层)中出现
如果一个白格 \(i\) 同时属于两个黑格组且位于第一层,且边 \(<i_0,i_1>\) 没有被割掉,那么显然这两个黑格组的第一层限制都必然不会被满足了,这时即使不同的黑格组之间互相影响也不会对最终的答案造成影响。第三层同理
而这时候如果 \(\min(w_x,w_y)\) 不在第二层而在第一层,那么假设这时候 \(i\) 位于第二层,有两个黑格组 \(X\) 和 \(Y\) ,\(X\) 只割掉了第三层,\(Y\) 只割掉了第一层,那么由于保留了边 \(<i_0,i_1>\) 之后 \(X\) 和 \(Y\) 的第二层能互通,故这时源点到汇点有一条 \(S\rightarrow X_1\rightarrow X_2\rightarrow Y_2\rightarrow Y_3\rightarrow T\) 的路径,这时就不合法了
Code
#include <bits/stdc++.h>
template <class T>
inline void read(T &res)
{
res = 0; bool bo = 0; char c;
while (((c = getchar()) < '0' || c > '9') && c != '-');
if (c == '-') bo = 1; else res = c - 48;
while ((c = getchar()) >= '0' && c <= '9')
res = (res << 3) + (res << 1) + (c - 48);
if (bo) res = ~res + 1;
}
template <class T>
inline T Min(const T &a, const T &b) {return a < b ? a : b;}
const int N = 1e5 + 10, M = 3e5 + 5, L = 4e6 + 5, INF = 0x3f3f3f3f;
int n, X[N], Y[N], w[N], ecnt = 1, nxt[L], adj[M], go[L], cap[L], len, que[M],
lev[M], cur[M], S, T, ans;
std::map<int, int> is[N];
int which(int x, int op) {return (x << 1) + op;}
void add_edge(int u, int v, int w)
{
nxt[++ecnt] = adj[u]; adj[u] = ecnt; go[ecnt] = v; cap[ecnt] = w;
nxt[++ecnt] = adj[v]; adj[v] = ecnt; go[ecnt] = u; cap[ecnt] = 0;
}
bool bfs()
{
for (int i = S; i <= T; i++) lev[i] = -1, cur[i] = adj[i];
lev[que[len = 1] = S] = 0;
for (int i = 1; i <= len; i++)
{
int u = que[i];
for (int e = adj[u], v = go[e]; e; e = nxt[e], v = go[e])
if (cap[e] && lev[v] == -1)
{
lev[que[++len] = v] = lev[u] + 1;
if (v == T) return 1;
}
}
return 0;
}
int dinic(int u, int flow)
{
if (u == T) return flow;
int res = 0, delta;
for (int &e = cur[u], v = go[e]; e; e = nxt[e], v = go[e])
if (cap[e] && lev[u] < lev[v])
{
delta = dinic(v, Min(cap[e], flow - res));
if (delta)
{
cap[e] -= delta; cap[e ^ 1] += delta;
res += delta; if (res == flow) break;
}
}
if (res < flow) lev[u] = -1;
return res;
}
void add(int i, int j, int op)
{
if (op) add_edge(S, which(j, 0), INF), add_edge(which(j, 1), which(i, 0), INF);
else add_edge(which(i, 1), which(j, 0), INF), add_edge(which(j, 1), T, INF);
}
int main()
{
read(n); read(n); read(n);
for (int i = 1; i <= n; i++)
read(X[i]), read(Y[i]), read(w[i]), is[X[i]][Y[i]] = i;
S = 1; T = n + 1 << 1;
for (int i = 1; i <= n; i++)
{
if (X[i] % 4 != (Y[i] + 1) % 2 * 2 + 1 && X[i] % 4 != Y[i] % 2 * 2)
add_edge(which(i, 0), which(i, 1), w[i]);
if (X[i] % 4 != (Y[i] + 1) % 2 * 2 + 1) continue;
int x = X[i], y = Y[i], j;
if (!(j = is[x + 1][y])) continue;
add_edge(which(i, 0), which(i, 1), Min(w[i], w[j]));
if (j = is[x - 1][y]) add(i, j, y & 1);
if (j = is[x][y - 1]) add(i, j, y & 1);
if (j = is[x][y + 1]) add(i, j, y & 1);
if (j = is[x + 2][y]) add(i, j, !(y & 1));
if (j = is[x + 1][y - 1]) add(i, j, !(y & 1));
if (j = is[x + 1][y + 1]) add(i, j, !(y & 1));
}
while (bfs()) ans += dinic(S, INF);
return std::cout << ans << std::endl, 0;
}
[LOJ#3022][网络流]「CQOI2017」老 C 的方块的更多相关文章
- @loj - 3022@ 「CQOI2017」老 C 的方块
目录 @description@ @solution@ @accepted code@ @details@ @description@ 老 C 是个程序员. 作为一个懒惰的程序员,老 C 经常在电脑上 ...
- 【LOJ】#3020. 「CQOI2017」小 Q 的表格
#3020. 「CQOI2017」小 Q 的表格 这个的话求出来\(g = gcd(a,b)\) 会修改所有gcd为g的位置 我们要求\((g,g)\)这个位置的数一定是\(g^{2}\)的倍数 之后 ...
- [LOJ#6437][BZOJ5373]「PKUSC2018」PKUSC
[LOJ#6437][BZOJ5373]「PKUSC2018」PKUSC 试题描述 九条可怜是一个爱玩游戏的女孩子. 最近她在玩一个无双割草类的游戏,平面上有 \(n\) 个敌人,每一个敌人的坐标为 ...
- 【LOJ】#2210. 「HNOI2014」江南乐
LOJ#2210. 「HNOI2014」江南乐 感觉是要推sg函数 发现\(\lfloor \frac{N}{i}\rfloor\)只有\(O(\sqrt{N})\)种取值 考虑把这些取值都拿出来,能 ...
- 【LOJ】#3098. 「SNOI2019」纸牌
LOJ#3098. 「SNOI2019」纸牌 显然选三个以上的连续牌可以把他们拆分成三个三张相等的 于是可以压\((j,k)\)为有\(j\)个连续两个的,有\(k\)个连续一个的 如果当前有\(i\ ...
- 【LOJ】#3103. 「JSOI2019」节日庆典
LOJ#3103. 「JSOI2019」节日庆典 能当最小位置的值一定是一个最小后缀,而有用的最小后缀不超过\(\log n\)个 为什么不超过\(\log n\)个,看了一下zsy的博客.. 假如\ ...
- 【LOJ】#3102. 「JSOI2019」神经网络
LOJ#3102. 「JSOI2019」神经网络 首先我们容易发现就是把树拆成若干条链,然后要求这些链排在一个环上,同一棵树的链不相邻 把树拆成链可以用一个简单(但是需要复杂的分类讨论)的树背包实现 ...
- 【LOJ】#3101. 「JSOI2019」精准预测
LOJ#3101. 「JSOI2019」精准预测 设0是生,1是死,按2-sat连边那么第一种情况是\((t,x,1) \rightarrow (t + 1,y,1)\),\((t + 1,y, 0) ...
- 【LOJ】#3097. 「SNOI2019」通信
LOJ#3097. 「SNOI2019」通信 费用流,有点玄妙 显然按照最小路径覆盖那题的建图思路,把一个点拆成两种点,一种是从这个点出去,标成\(x_{i}\),一种是输入到这个点,使得两条路径合成 ...
随机推荐
- 2018百度之星资格赛A B F
A.调查问卷 度度熊为了完成毕业论文,需要收集一些数据来支撑他的论据,于是设计了一份包含 mm 个问题的调查问卷,每个问题只有 'A' 和 'B' 两种选项. 将问卷散发出去之后,度度熊收到了 nn ...
- 理解Servlet
题记:框架横行,似乎已经忘记JavaWeb最基础Servlet是如何工作的,这也是为什么要写这篇文章. Servlet是Java语言应用到Web的扩展技术,是运行在Web应用服务器上的Java程序.与 ...
- C#面试题整理(不带答案)
1.维护数据库的完整性.一致性.你喜欢用触发器还是自写业务逻辑?为什么? 2.什么是事务?什么是锁? 3.什么是索引,有什么优点? 4.视图是什么?游标是什么? 5.什么是存储过程?有什么优 ...
- Qt4与Qt3的主要不同
Qt4与Qt3的主要不同 1)QT4 中提供了大量新控件,虽然它也保持了旧的控件,并命名为Qt3XXX,但是这样的控件没准在今后的哪个QT版本中就不被支持了,所以还是换吧,控件替换的 工作是巨大的,这 ...
- 【Docker】Ubuntu16.04将Docker升级至最新版
1.使用curl升级到最新版 curl -fsSL https://get.docker.com/ | sh 2.重启Docker sudo systemctl restart docker 3.设置 ...
- Ubuntu安装微信,解决deepin“版本过低”或NO_PUBKEY问题
在搜索引擎搜索Ubuntu安装微信,最多的结果是通过deepin安装 但是里面使用的deepin-for-ubuntu 安装之后微信扫码会提示版本过低 直接安装deepin.com.wechat_2. ...
- h5 页面 实现单选题,多选题功能。
效果图: 项目要求: 1:实现单选题和多选题区分 (这个根据后端传来的数据判断 ) 2 单选选中效果 和 多选选中效果(利用input 和label ) 3.答题成功与失败 分单选和多选的情况 ...
- Visio流程图表
基本流程图: 流程图类别 基本流程图的四种类型 打开基本流程图 注意页面内引用跟跨页引用 就是两个按钮的作用 就是一个按钮的作用 点击跳转 按钮设置好之后可以输入数字 方便区分跳转 下面是跨职能流程图 ...
- 定位布局中关于z-index的一些问题
定位布局中关于z-index的一些问题 使不同父元素的子元素不会被其他父元素遮盖 背景 两父元素相互遮盖(或部分遮盖) html如下 <div class="main"> ...
- 前端-jQuery介绍
目录 jQuery介绍 jQuery的优势 jQuery内容: jQuery版本 jQuery对象 jQuery基础语法 查找标签 基本选择器 层级选择器: 基本筛选器: 属性选择器: 表单筛选器: ...