[bzoj1067][SCOI2007]降雨量——线段树+乱搞

题目大意

传送门

题解

我国古代有一句俗话。

骗分出奇迹,乱搞最神奇!

这句话在这道题上得到了鲜明的体现。
我的方法就是魔改版线段树,乱搞搞一下,首先借鉴了黄学长的建树方法,直接用一个节点维护年份的区间,但是这样就带来了问题,就是在查询的时候非常难以操作。经过不断的乱搞,我终于把正确的操作方式搞了出来。。。
另外这个题细节还是很多。详见胡泽聪dalao的分析

代码

#include <bits/stdc++.h>
using namespace std;
const int maxn = 50005;
struct seg {
  int l, r, mx, know;
} t[maxn * 4];
int s[maxn], n, m;
void build(int k, int l, int r) {
  if (l == r) {
    scanf("%d %d", &t[k].l, &t[k].mx);
    t[k].r = t[k].l;
    t[k].know = 1;
    return;
  }
  int mid = (l + r) >> 1;
  build(k << 1, l, mid);
  build(k << 1 | 1, mid + 1, r);
  t[k].know = (t[k << 1].know && t[k << 1 | 1].know);
  if (t[k << 1].r + 1 != t[k << 1 | 1].l)
    t[k].know = 0;
  t[k].l = t[k << 1].l;
  t[k].r = t[k << 1 | 1].r;
  t[k].mx = max(t[k << 1].mx, t[k << 1 | 1].mx);
}
int query_mx(int k, int x, int y) {
  int l = t[k].l, r = t[k].r;
  if (y < x)
    return 0;
  if (x < l)
    x = l;
  if (y > r)
    y = r;
  if (x == l && r == y)
    return t[k].mx;
  int mid = t[k << 1].r;
  int nxmid = t[k << 1 | 1].l;
  if (y < l || x > r)
    return 0;
  int ans = -0x3f3f3f;
  if (x >= l && y < nxmid && x <= mid) {
    ans = max(ans, query_mx(k << 1, x, y));
  } else if (y <= r && x > mid && y >= nxmid) {
    ans = max(ans, query_mx(k << 1 | 1, x, y));
  } else if (x >= l && x <= mid && y >= nxmid && y <= r) {
    ans = max(ans, query_mx(k << 1, x, mid));
    ans = max(ans, query_mx(k << 1 | 1, nxmid, y));
  } else
    return 0;
  return ans;
}
int query_know(int k, int x, int y) {
  int l = t[k].l, r = t[k].r;
  if (y < x)
    return 0;
  if (x < l || y > r)
    return 0;
  if (x == l && y == r)
    return t[k].know;
  else if (r - l == 0)
    return 0;
  bool ans = true;
  int mid = t[k << 1].r;
  int nxmid = t[k << 1 | 1].l;
  if (x >= l && y < nxmid && x <= mid) {
    ans = ans && query_know(k << 1, x, y);
  } else if (y <= r && x > mid && y >= nxmid) {
    ans = ans && query_know(k << 1 | 1, x, y);
  } else if (x >= l && x <= mid && y >= nxmid && y <= r) {
    ans = ans && query_know(k << 1, x, mid);
    ans = ans && query_know(k << 1 | 1, nxmid, y);
    ans = ans && (nxmid-mid <= 1) ?  1 : 0;
  } else
    return 0;
  return ans;
}
int query(int k, int x) {
  if (t[k].l == t[k].r)
    return t[k].mx;
  if (x <= t[k << 1].r)
    return query(k << 1, x);
  else if (x >= t[k << 1 | 1].l)
    return query(k << 1 | 1, x);
  else
    return -1;
}
int main() {
  //freopen("rain.in", "r", stdin);
  //freopen("rain.ans", "w", stdout);
  scanf("%d", &n);
  build(1, 1, n);
  scanf("%d", &m);
  // cout << query_mx(1, 2003, 2007);
   //cout << query_know(1, -138, -129) << endl;
  // cout << query_know(1, 2004, 2006) << endl;
  for (int i = 1; i <= m; i++) {
    int x, y;
    scanf("%d %d", &x, &y);
    if (query_know(1, x, y) && (query(1, x) >= query(1, y)) &&
        (query_mx(1, x + 1, y - 1) < query(1, y)))
      printf("true
");
    else if (query_know(1, x, x) && query_know(1, y, y) &&
             !(query_know(1, x + 1, y - 1)) &&
             (query_mx(1, x + 1, y - 1) < query(1, y)) &&
             (query(1, x) >= query(1, y)))
      printf("maybe
");
    else if (query_know(1, x, x) && !(query_know(1, y, y)) && (query_mx(1,x+1,y-1) < query(1,x)))
      printf("maybe
");
    else if (query_know(1, y, y) && !(query_know(1, x, x)) &&
             (query_mx(1, x + 1, y - 1) < query(1, y)))
      printf("maybe
");
    else if ((!(query_know(1, x, x)) && (!(query_know(1, y, y)))))
      printf("maybe
");
    else
      printf("false
");
  }
  return 0;
}

Ps.开始对着样例乱搞出的代码居然得了50分

原文地址:https://www.cnblogs.com/gengchen/p/6438796.html