@loj - [email protected] 「CQOI2017」老 C 的方块

目录

  • @[email protected]
  • @[email protected]
  • @accepted [email protected]
  • @[email protected]

@[email protected]

老 C 是个程序员。

作为一个懒惰的程序员,老 C 经常在电脑上玩方块游戏消磨时间。游戏被限定在一个由小方格排成的 R 行 C 列网格上,如果两个小方格有公共的边,就称它们是相邻的,而且有些相邻的小方格之间的公共边比较特殊。
特殊的公共边排列得有很强的规律。首先规定,第 1 行的前两个小方格之间的边是特殊边。然后,特殊边在水平方向上每 4 个小方格为一个周期,在竖直方向上每 2 个小方格为一个周期。所有的奇数列与下一列之间都有特殊边,且所在行的编号从左到右奇偶交替。

下图所示是一个 R = C = 8 的网格,蓝色标注的边是特殊边。
首先,在第 1 行,第 1 列和第 2 列之间有一条特殊边。因为竖直方向周期为 2,所以所有的奇数行,第 1 列和第 2 列之间都有特殊边。因为水平方向周期为 4,所以所有奇数行的第 5 列和第 6 列之间也有特殊边,如果网格足够大,所有奇数行的第 9 列和第 10 列、第 13 列和第 14 列之间都有特殊边。
因为所有的奇数列和下一列之间都有特殊边,所以第 3 列和第 4 列、第 7 列和第 8 列之间也有特殊边,而所在行的编号从左到右奇偶交替,所以它们的特殊边在偶数行。如果网格的规模更大,我们可以用同样的方法找出所有的特殊边。

网格的每个小方格刚好可以放入一个小方块,在游戏的一开始,有些小方格已经放上了小方块,另外的小方格没有放。
老 C 很讨厌下图所示的图形,如果他发现有一些小方块排列成了它讨厌的形状(特殊边的位置也要如图中所示),就很容易弃疗,即使是经过任意次旋转、翻转后排列成讨厌的形状,老 C 也同样容易弃疗。

为了防止弃疗,老 C 决定趁自己还没有弃疗,赶紧移除一些格子里小方块,使得剩下的小方块不能构成它讨厌的形状。但是游戏里每移除一个方块都是要花费一些金币的,每个方块需要花费的金币有多有少参差不齐。老 C 当然希望尽可能少的使用游戏里的金币,但是最少要花费多少金币呢?老 C 懒得思考,就把这个问题交给你了。

输入格式
第一行有 3 个正整数 C, R, n,表示 C 列 R 行的网格中,有 n 个小方格放了小方块。
接下来 n 行,每行 3 个正整数 x, y, w,表示在第 x 列第 y 行的小方格里放了小方块,移除它需要花费 w 个金币。保证不会重复,且都在网格范围内。

输出格式
输出一行,包含一个整数,表示最少花费的金币数量。

样例
样例输入 1
2 2 4
1 1 5
1 2 6
2 1 7
2 2 8
样例输入 1
5
样例输入 2
3 3 7
1 1 10
1 2 15
1 3 10
2 1 10
2 2 10
2 3 10
3 1 10
样例输出 2
15
数据范围与提示
1 <= C, R, n <= 10^5, 1 <= w <= 10^4。

@[email protected]

题意绕晕人.jpg。
但实际上有用的信息全在图片里面 2333。

假如第 x 列第 y 行的 (x, y) 与 (x + 1, y) 同时存在,且它们之间有特殊边(即图片上的蓝色边),则必须要从如下的几种选择中选择一种:
(1)删除 (x, y) 或 (x + 1, y)(这个地方显然是选择删除费用较小的那个)。
(2)删除 (x, y) 周围除了 (x + 1, y) 以外相邻的 3 个 (x - 1, y), (x, y - 1), (x, y + 1)(如果它们存在)。
(3)删除 (x + 1, y) 周围除了 (x, y) 以外相邻的 3 个 (x + 2, y), (x + 1, y - 1), (x + 1, y + 1)(如果它们存在)。

“必须从几种选择中选择一种” 并且求最小花费,联想到最小割。

考虑将那些与特殊边不相邻的点拿出来,进行黑白染色(这里的染色不是平常的相邻染成异色,而是上面所说的 (x, y) 周围的三个点染成同色,(x + 1, y) 周围三个点染成同色,且 (x, y) 与 (x + 1, y) 周围的点为异色)。
将源点连黑点,容量为题目中所说的 w;将白点连汇点,容量为题目中所说的 w。

假如第 x 列第 y 行的 (x, y) 与 (x + 1, y) 同时存在,则 (x, y) 和 (x + 1, y) 之间连边,容量为 min(w1, w2)(分别表示两个格子删除的费用),方向看它们周围的格子颜色是什么:如果 (x, y) 与黑色相邻则 (x, y) -> (x + 1, y);否则 (x + 1, y) -> (x, y)。
然后周围的黑点向它相邻的点连,它们向周围的白点连边,容量都为 inf。

可以发现这样连出来是可以满足我们所需要的限制。
不要问我为什么 10^5 跑得过,问就是形而上学。

@accepted [email protected]

#include<map>
#include<cstdio>
#include<iostream>
using namespace std;
typedef pair<int, int> pii;
const int MAXN = 100000;
const int MAXV = 2*MAXN;
const int MAXE = 20*MAXV;
const int INF = int(1E9);
const pii d1[] = {make_pair(0, 1), make_pair(0, -1), make_pair(1, 0)};
const pii d2[] = {make_pair(0, 1), make_pair(0, -1), make_pair(-1, 0)};
struct Graph{
    struct edge{
        edge *nxt, *rev;
        int to, cap, flow;
    }edges[MAXE + 5], *adj[MAXV + 5], *ecnt;
    int s, t, d[MAXV + 5], vd[MAXV + 5];
    void init() {ecnt = &edges[0];}
    void addedge(int u, int v, int c) {
        edge *p = (++ecnt), *q = (++ecnt);
        p->to = v, p->cap = c, p->flow = 0;
        p->nxt = adj[u], adj[u] = p;
        q->to = u, q->cap = 0, q->flow = 0;
        q->nxt = adj[v], adj[v] = q;
        p->rev = q, q->rev = p;
//      printf("! %d %d %d\n", u, v, c);
    }
    int aug(int x, int tot) {
        if( x == t ) return tot;
        int sum = 0, mind = t + 1;
        for(edge *p=adj[x];p;p=p->nxt) {
            if( p->cap != p->flow ) {
                if( d[p->to] + 1 == d[x] ) {
                    int del = aug(p->to, min(tot - sum, p->cap - p->flow));
                    p->flow += del, p->rev->flow -= del, sum += del;
                    if( sum == tot || d[s] == t + 1 ) return sum;
                }
                mind = min(mind, d[p->to]);
            }
        }
        if( sum == 0 ) {
            vd[d[x]]--;
            if( vd[d[x]] == 0 ) {
                d[s] = t + 1;
                return sum;
            }
            d[x] = mind + 1;
            vd[d[x]]++;
        }
        return sum;
    }
    int max_flow(int _s, int _t) {
        s = _s, t = _t;
        int flow = 0;
        while( d[s] != t + 1 )
            flow += aug(s, INF);
        return flow;
    }
}G;
map<pii, int>mp;
int C, R, n;
pii p[MAXN + 5]; int w[MAXN + 5];
int func(pii p) {
    if( p.second & 1 ) return (p.first - 1) % 4;
    else return (p.first - 1) % 4 + 4;
}
int main() {
    scanf("%d%d%d", &C, &R, &n), G.init();
    for(int i=1;i<=n;i++)
        scanf("%d%d%d", &p[i].first, &p[i].second, &w[i]), mp[p[i]] = i;
    int s = n + 1, t = n + 2;
    for(int i=1;i<=n;i++) {
        int type = func(p[i]);
        if( type == 4 || type == 3 )
            G.addedge(s, i, w[i]);
        if( type == 5 || type == 2 )
            G.addedge(i, t, w[i]);
        if( type == 0 ) {
            if( mp.count(make_pair(p[i].first + 1, p[i].second)) ) {
                int j = mp[make_pair(p[i].first + 1, p[i].second)];
                G.addedge(i, j, min(w[i], w[j]));
                for(int k=0;k<3;k++)
                    if( mp.count(make_pair(p[j].first + d1[k].first, p[j].second + d1[k].second)) )
                        G.addedge(j, mp[make_pair(p[j].first + d1[k].first, p[j].second + d1[k].second)], INF);
                for(int k=0;k<3;k++)
                    if( mp.count(make_pair(p[i].first + d2[k].first, p[i].second + d2[k].second)) )
                        G.addedge(mp[make_pair(p[i].first + d2[k].first, p[i].second + d2[k].second)], i, INF);
            }
        }
        if( type == 6 ) {
            if( mp.count(make_pair(p[i].first + 1, p[i].second)) ) {
                int j = mp[make_pair(p[i].first + 1, p[i].second)];
                G.addedge(j, i, min(w[i], w[j]));
                for(int k=0;k<3;k++)
                    if( mp.count(make_pair(p[j].first + d1[k].first, p[j].second + d1[k].second)) )
                        G.addedge(mp[make_pair(p[j].first + d1[k].first, p[j].second + d1[k].second)], j, INF);
                for(int k=0;k<3;k++)
                    if( mp.count(make_pair(p[i].first + d2[k].first, p[i].second + d2[k].second)) )
                        G.addedge(i, mp[make_pair(p[i].first + d2[k].first, p[i].second + d2[k].second)], INF);
            }
        }
    }
    printf("%d\n", G.max_flow(s, t));
}

@[email protected]

注意!它题目中说的是!第 x 列第 y 行!!!
被这个题意卡到自闭。以为自己读懂了结果样例一看好像不大对。。。

原文地址:https://www.cnblogs.com/Tiw-Air-OAO/p/11416627.html

时间: 2024-11-02 13:35:54

@loj - [email protected] 「CQOI2017」老 C 的方块的相关文章

@loj - [email&#160;protected]「CEOI2017」Building Bridges

目录 @[email protected] @[email protected] @accepted [email protected] @[email protected] @[email protected] 有 n 根柱子依次排列,第 i 根柱子的高度为 hi .现可以花费 (hi - hj)^2 的代价建桥架在第 i 根柱子和第 j 根柱子之间. 所有用不到的柱子都会被拆除,第 i 根柱子被拆除的代价为 wi . 求用桥把第 1 根柱子和第 n 根柱子连接的最小代价.注意桥梁不能在端点以

@loj - [email&#160;protected]「SDOI2014」Lis

目录 @[email protected] @[email protected] @accepted [email protected] @[email protected] @[email protected] 给定序列 A,序列中的每一项 Ai 有删除代价 Bi 和附加属性 Ci 请删除若干项,使得 A 的最长上升子序列长度减少至少 1,且付出的代价之和最小,并输出方案. 如果有多种方案,请输出将删去项的附加属性排序之后,字典序最小的一种. 输入格式 输入包含多组数据. 输入的第一行包含整数

@loj - [email&#160;protected] 「SDOI2017」硬币游戏

目录 @[email protected] @[email protected] @accepted [email protected] @[email protected] @[email protected] 周末同学们非常无聊,有人提议,咱们扔硬币玩吧,谁扔的硬币正面次数多谁胜利. 大家纷纷觉得这个游戏非常符合同学们的特色,但只是扔硬币实在是太单调了. 同学们觉得要加强趣味性,所以要找一个同学扔很多很多次硬币,其他同学记录下正反面情况. 用 H 表示正面朝上, 用 T 表示反面朝上,扔很多

@loj - [email&#160;protected] 「CTSC2016」时空旅行

目录 @[email protected] @[email protected] @accepted [email protected] @[email protected] @[email protected] 2045 年,人类的技术突飞猛进,已经找到了进行时空旅行的方法.小 R 得到了一台时空旅行仪,他想用它调查不同时空中人类的发展状况. 根据平行时空理论,宇宙中存在着很多独立的时空,每个时空在下一个时间点还会分化出若干个不同的时空.宇宙是一个三维空间,人类使用空间直角坐标系来描述空间中的

@loj - [email&#160;protected] 「清华集训 2017」生成树计数

目录 @[email protected] @[email protected] @正文@ @补充@ @accepted [email protected] @[email protected] @[email protected] 在一个 s 个点的图中,存在 s - n 条边,使图中形成了 n 个连通块,第 i 个连通块中有 \(a_i\) 个点. 现在我们需要再连接 n - 1 条边,使该图变成一棵树.对一种连边方案,设原图中第 i 个连通块连出了 \(d_i\) 条边,那么这棵树 T 的

「珍藏」老司机为你推荐10个炫酷的开源库,看完的人都收藏了

前言 技术群里面经常有人问到一些炫酷的UI效果实现方法,有时候我都是给一个相同或者相似效果的Github链接,有同学私信给我说,大佬,怎么这些效果你都能找到?你是怎么搜索的,或者有其他什么秘方?会利用Google.百度等搜索工具搜索是一方面,另一个重要的方面是:记录搜藏,当看到一个炫酷的效果的时候,记得收藏起来,记录到自己云笔记或者收藏夹里,看得多了,印象就比较深刻,当遇到类似效果的时候,到自己记录收藏的地方找就是了.今天为大家推荐我所收藏的一些炫酷实用的效果的开源库(选择其中10个). 1.D

「考试」老司机的狂欢

啊考场上没想到. 直接二分答案,然后$nlogn$求解最长上升序列来$check$是否大于$K$即可. 然后恶心的是要求输出方案,而且...字典序最小. 我们考虑二分出答案之后求出方案. $LIS$的过程其实类似于建树,我们要把当前的决策挂在当前树上某一深度的点中,字典序最小的方案下面. 那么当我们比较两个方案的时候,只需要求出他们在$LIS$树上的$LCA$,那么在$LCA$以上的部分完全相同,所以只需要比较他们在$LCA$一下的部分中最小值的大小,较小的更优,一边跑$LIS$一边跑倍增$LC

loj#2552. 「CTSC2018」假面

题目链接 loj#2552. 「CTSC2018」假面 题解 本题严谨的证明了我菜的本质 对于砍人的操作好做找龙哥就好了,blood很少,每次暴力维护一下 对于操作1 设\(a_i\)为第i个人存活的概率,\(d_i\)为死掉的概率,\(g_{i,j}\)是除i以外活了j个人的概率 那个选中i人的答案就是 \[a_i\times\sum_{j = 0} ^{k - 1}\frac{g_{i,j}}{j + 1}\] 对于\(g_{i,j}\) ,设\(f_{i,j}\)表示前\(i\)个人有\(

loj#2076. 「JSOI2016」炸弹攻击 模拟退火

目录 题目链接 题解 代码 题目链接 loj#2076. 「JSOI2016」炸弹攻击 题解 模拟退火 退火时,由于答案比较小,但是温度比较高 所以在算exp时最好把相差的点数乘以一个常数让选取更差的的概率降低 代码 #include<ctime> #include<cmath> #include<cstdio> #include<cstring> #include<algorithm> #define gc getchar() #define