bzoj1067 [SCOI2007]降雨量 (线段树)

Description

我们常常会说这样的话:“X年是自Y年以来降雨量最多的”。它的含义是X年的降雨量不超过Y年,且对于任意Y<Z<X,Z年的降雨量严格小于X年。例如2002,2003,2004和2005年的降雨量分别为4920,5901,2832和3890,则可以说“2005年是自2003年以来最多的”,但不能说“2005年是自2002年以来最多的”由于有些年份的降雨量未知,有的说法是可能正确也可以不正确的。

Input

输入仅一行包含一个正整数n,为已知的数据。以下n行每行两个整数yi和ri,为年份和降雨量,按照年份从小到大排列,即yi<yi+1。下一行包含一个正整数m,为询问的次数。以下m行每行包含两个数Y和X,即询问“X年是自Y年以来降雨量最多的。”这句话是必真、必假还是“有可能”。

Output

对于每一个询问,输出true,false或者maybe。

Sample Input

6
2002 4920
2003 5901
2004 2832
2005 3890
2007 5609
2008 3024
5
2002 2005
2003 2005
2002 2007
2003 2007
2005 2008

Sample Output

false
true
false
maybe
false

HINT

100%的数据满足:1<=n<=50000, 1<=m<=10000, -10^9<=yi<=10^9, 1<=ri<=10^9

题解

维护区间最大值,注意好每种情况的特判。

代码

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;

const int N = 1000010;

struct Node {
    int l, r, maxx;
}t[N];
int n, m, x, y, len, nowmax, root, year[N], rain[N];

void build(int root, int L, int R) {
    t[root].l = L; t[root].r = R;
    if(L == R) {
        t[root].maxx = rain[L];
        return ;
    }
    int mid = (L + R) >> 1;
    build(root << 1, L, mid);
    build(root << 1 | 1, mid + 1, R);
    t[root].maxx = max(t[root << 1].maxx, t[root << 1 | 1].maxx);
    return ;
}

int query(int root, int L, int R) {
    int l = t[root].l, r = t[root].r;
    if(L <= l && r <= R)
        return t[root].maxx;
    int rt = 0;
    int mid = (l + r) >> 1;
    if(L <= mid)
        rt = max(rt, query(root << 1, L, R));
    if(R > mid)
        rt = max(rt, query(root << 1 | 1, L, R));
    return rt;
}

int check(int x, int y) {
    if(y < x) return 0;
    if(y == x) return 2;
    int a = upper_bound(year + 1, year + n + 1, x) - year - 1;
    int b = upper_bound(year + 1, year + n + 1, y) - year - 1;
    len = y - x; nowmax = query(1, a + 1, b - 1);

    if(b <= 1 || a == n || (year[a] != x && year[b] != y)) return 2;
    if(len == b - a && rain[a] >= rain[b] && year[a] == x && year[b] == y && nowmax < rain[b]) return 1;

    if(year[a] != x) {
        if(nowmax >= rain[b]) return 0;
        else return 2;
    }
    if(year[b] != y) {
        if(query(1, a + 1, b) >= rain[a]) return 0;
        else return 2;
    }
    if(nowmax < rain[b] && rain[b] <= rain[a]) {
        if(len == b - a) return 1;
        else return 2;
    }
    return 0;
}

int main() {
    scanf("%d", &n);
    for(int i = 1; i <= n; i ++)
        scanf("%d%d", &year[i], &rain[i]);
    build(root = 1, 1, n);
    scanf("%d", &m);
    while(m --) {
        scanf("%d%d", &x, &y);
        int q = check(x, y);
        if(q == 2) printf("maybe\n");
        else if(q == 0) printf("false\n");
        else printf("true\n");
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值