【NOI2018模拟4.3】paint

标签: 信息学 greedy 线段树 单调栈
36人阅读 评论(0) 收藏 举报
分类:

Description:

这里写图片描述
1<=n<=100000

题解:

有一个非常易证但是不显然的结论。

答案矩形一定过横轴中线或数轴中线。

只考虑数轴中线先,横轴中线翻转一下再做就行了。

中线把点分成了两个部分。

按x排个序,x相同的只取最接近中线的一个。

然后扫,建单调栈,正常的暴力的话每次查询是单调栈的长度的复杂度。

但是可以离散+线段树。

线段树存的是以这个坐标为左边界,往上下的最长长度,便于统计,还要减去一个横坐标。

细节非常繁琐。

Code:

#include<cstdio>
#include<cstring>
#include<algorithm>
#define fo(i, x, y) for(int i = x; i <= y; i ++)
using namespace std;

const int N = 2e5 + 5;

int w, h, n, ans, num[N];
struct node {
    int x, y;
    node (int _x = 0, int _y = 0) {x = _x, y = _y;}
} a[N], b[N];


int pl, pr, px;
struct tree {
    int mx, z;
} t[N * 4];
void down(int i) {
    if(t[i].z) {
        t[i + i].z += t[i].z; t[i + i].mx += t[i].z;
        t[i + i + 1].z += t[i].z; t[i + i + 1].mx += t[i].z;
        t[i].z = 0;
    }
}
void add(int i, int x, int y) {
    if(y < pl || x > pr) return;
    if(x >= pl && y <= pr) {t[i].z += px; t[i].mx += px;return;}
    int m = x + y >> 1; down(i);
    add(i + i, x, m); add(i + i + 1, m + 1, y);
    t[i].mx = max(t[i + i].mx, t[i + i + 1].mx);

}
void cc(int x, int y, int c) {
    pl = x, pr = y, px = c; add(1, 1, n + 2);
}

int z1[N], z2[N], d1, d2, z;

bool cmp(node a, node b) {return a.x < b.x;}

int b1[N], b2[N];

void GG() {
    memset(b1, 0, sizeof b1);
    memset(b2, 0, sizeof b2);
    z = h / 2;
    a[n + 1] = node(0, z + 1); a[n + 2] = node(0, z); a[n + 3] = node(w, h);
    sort(a + 1, a + n + 4, cmp);
    memset(t, 0, sizeof t);
    int tot = 0;
    fo(i, 1, n + 3) {
        if(i == 1 || a[i].x != a[i - 1].x)
            ++ tot, cc(tot, tot, h - a[i].x);
        num[i] = tot;
    }
    fo(i, 1, n + 3)
        if(a[i].y > z) {
            if(!b1[num[i]]) b1[num[i]] = i; else {
                if(a[i].y < a[b1[num[i]]].y)
                    b1[num[i]] = i;
            }
        } else {
            if(!b2[num[i]]) b2[num[i]] = i; else {
                if(a[i].y > a[b2[num[i]]].y)
                    b2[num[i]] = i;
            }
        }
    d1 = d2 = 1;
    fo(i, 1, n + 3) {
        if(!a[i].x && a[i].y == z) z1[1] = i;
        if(!a[i].x && a[i].y == z + 1) z2[1] = i;
    }
    fo(i, 1, n + 3) {
        if(!a[i].x && (a[i].y == z || a[i].y == z + 1))
            continue;
        if(a[i].y > z) {
            if(i != b1[num[i]]) continue;
        } else {
            if(i != b2[num[i]]) continue;
        }
        ans = max(ans, t[1].mx + a[i].x);
        if(a[i].y > z) {
            cc(num[z1[d1]], num[i] - 1, a[i].y - h);
            while(a[z1[d1]].y > a[i].y) {
                cc(num[z1[d1 - 1]], num[z1[d1]] - 1, a[i].y - a[z1[d1]].y);
                d1 --;
            }
            z1[++ d1] = i;
        } else {
            cc(num[z2[d2]], num[i] - 1, -a[i].y);
            while(a[z2[d2]].y < a[i].y) {
                cc(num[z2[d2 - 1]], num[z2[d2]] - 1, a[z2[d2]].y - a[i].y );
                d2 --;
            }
            z2[++ d2] = i;
        }
    }
    return;
}

int main() {
    freopen("paint.in", "r", stdin);
    freopen("paint.out", "w", stdout);
    scanf("%d %d %d", &w, &h, &n);
    fo(i, 1, n) scanf("%d %d", &a[i].x, &a[i].y);
    fo(i, 1, n) b[i] = a[i];
    GG();
    swap(w, h); fo(i, 1, n) a[i] = node(b[i].y, b[i].x);
    GG();
    printf("%d", ans * 2);
}
查看评论

ATM取款机模拟实战项目

-
  • 1970年01月01日 08:00

JZOJ 5627. 【NOI2018模拟4.3】paint

JZOJ 5627. 【NOI2018模拟4.3】paint DescriptionInputOutputSample Input样例输入110 10 4 1 6 4 1 6 9 9 4样例输...
  • liyizhixl
  • liyizhixl
  • 2018-04-04 09:16:40
  • 42

JZOJ5625. 【NOI2018模拟4.3】Max

题目 题解 这里的m也就是操作数很小,考虑状压操作数。 设gi,s,kgi,s,kg_{i,s,k}表示第i个数,操作状态为s,变成k的概率, 转移的时候就枚举哪一个操作还有加多少。 ...
  • lijf2001
  • lijf2001
  • 2018-04-08 12:16:08
  • 71

JZOJ5640. 【NOI2018模拟4.9】劈配

题目描述 轻车熟路的 Zayid 顺利地通过了海选,接下来的环节是导师盲选,这一阶段的规则 是这样的: 总共 n 名参赛选手(编号从 1 至 n)每人写出一份代码并介绍自己的梦想。接着由 所有导师对...
  • lijf2001
  • lijf2001
  • 2018-04-10 10:36:15
  • 12

JZOJ 5628. 【NOI2018模拟4.4】Travel

JZOJ 5628. 【NOI2018模拟4.4】Travel Description有N个人出去旅行,第i个人去A国有Ai种游玩方式,去B国有Bi种游玩方式,问至少有C个人去A国的情况下,所有人的游...
  • liyizhixl
  • liyizhixl
  • 2018-04-04 15:54:37
  • 36

JZOJ5616. 【NOI2018模拟3.31】沧海尘记

问题描述 南阳渡旁有一个饿得发昏的道士,详聊过后,才知道他想要兜售身上的道书以求温 饱。但是他所开出的价码,实在让人不敢领教…… 买下道书后,第⼀⻚上的内容就令你⼤吃⼀惊—— 如果把⼈按照拥有财富...
  • lijf2001
  • lijf2001
  • 2018-03-31 22:21:53
  • 30

JZOJ5646. 【NOI2018模拟4.12】染色游戏

题目 题目大意 给出aiaia_i,要求从中选择一些aiaia_i出来,可以不选(一定要注意) 使得这些aiaia_i是严格上升。 对于每一段连续的空位,都要付出一些代价。 要求最大...
  • lijf2001
  • lijf2001
  • 2018-04-14 22:09:14
  • 11

JZOJ5623. 【NOI2018模拟4.2】program

题目 题目大意 给出一个操作序列,询问给出一个区间, 求这个区间操作所产生的答案。 题解 假设先对原操作序列求一次答案, 那么询问的某个区间的答案就一定是连续的一段。 有了这...
  • lijf2001
  • lijf2001
  • 2018-04-08 11:52:07
  • 26

[jzoj5635][CF578D]【NOI2018模拟4.7】LCS

三倍经验题当然要优先做啦~~~ jzoj5636 CF578D Solution T可以看成S去掉一个字母再加一个字母. 首先把S分成一些相同字母组成的段(假设有k段),去掉这些段中的...
  • FarmerJohnOfZS
  • FarmerJohnOfZS
  • 2018-04-11 21:02:34
  • 22

JZOJ5651. 【NOI2018模拟4.13】硬币游戏

分析 我们一组一组往里面加边,如果成环了就考虑枚举删掉哪一条边然后增广,然后就没了 代码 #include &amp;lt;bits/stdc++.h&amp;gt; #defin...
  • ypxrain
  • ypxrain
  • 2018-04-13 21:15:04
  • 9
    个人资料
    持之以恒
    等级:
    访问量: 7万+
    积分: 3154
    排名: 1万+
    友情链接
    文章分类
    最新评论