POJ 3264 Balanced Lineup

\(POJ\) \(3264\) \(Balanced\) \(Lineup\)

一、大致题意

给定一组序列编号从\(1-n\), 有\(q\)个询问, 每次询问\([l, r]\)区间的最大值与最小值的 差值.

二、解题思路

  • 单点修改构建数据结构
    • 比较适合树状数组,代码更简单
    • 如果使用线段树,那么不需要\(pushdown\)+懒标记优化,一切递归到底,慢就慢吧
  • 区间查询,求区间最大+最小值

三、树状数组解法

#include <algorithm>
#include <cstdio>
#include <cstring>

const int INF = 0x3f3f3f3f;
using namespace std;
const int N = 50010;

int a[N];
int n, q;

// 树状数组模板
int c1[N], c2[N]; // c1:维护最大值,c2:维护最小值

int lowbit(int x) {
    return x & -x;
}

void update(int x, int d) {
    for (int i = x; i < N; i += lowbit(i)) c1[i] = max(c1[i], d), c2[i] = min(c2[i], d);
}

int query(int x, int y) {
    int mx = 0, mi = INF;
    while (y >= x) {                            // 右侧到左侧
        mx = max(mx, a[y]), mi = min(mi, a[y]); // 每次最后侧的点参加评比

        for (--y; y - x >= lowbit(y); y -= lowbit(y)) // 从x~y-1检查每个片断
            mx = max(mx, c1[y]), mi = min(mi, c2[y]);
    }
    return mx - mi;
}

int main() {
#ifndef ONLINE_JUDGE
    freopen("POJ3264.in", "r", stdin);
#endif
    memset(c1, 0, sizeof c1);
    memset(c2, 0x3f, sizeof c2); // 最小值,需要先设置最大,最大值默认的就是0,不需要设置
    scanf("%d %d", &n, &q);

    for (int i = 1; i <= n; i++) {
        scanf("%d", &a[i]);
        update(i, a[i]);
    }

    while (q--) {
        int a, b;
        scanf("%d %d", &a, &b);
        printf("%d\n", query(a, b));
    }
    return 0;
}

四、线段树解法

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

const int INF = 0x3f3f3f3f;
const int N = 50010;
int a[N];

struct Node {
    int l, r, mx, mi;
} tr[N << 2];

int mx, mi;
void pushup(int u) {
    tr[u].mx = max(tr[u << 1].mx, tr[u << 1 | 1].mx);
    tr[u].mi = min(tr[u << 1].mi, tr[u << 1 | 1].mi);
}
void build(int u, int l, int r) {
    tr[u].l = l, tr[u].r = r; // POJ对于tr[u]={l,r}报编译错误,编译器版本太低,需要手动实现构造函数

    if (l == r) {
        tr[u].mx = a[l], tr[u].mi = a[l];
        return;
    }
    int mid = (l + r) >> 1;
    build(u << 1, l, mid), build(u << 1 | 1, mid + 1, r);
    pushup(u);
}

void query(int u, int l, int r) {
    if (l <= tr[u].l && r >= tr[u].r) {
        mx = max(mx, tr[u].mx), mi = min(mi, tr[u].mi);
        return;
    }

    int mid = (tr[u].l + tr[u].r) >> 1;

    if (l <= mid) query(u << 1, l, r);
    if (r > mid) query(u << 1 | 1, l, r);
}

int main() {
#ifndef ONLINE_JUDGE
    freopen("POJ3264.in", "r", stdin);
#endif
    // 加快读入
    ios::sync_with_stdio(false), cin.tie(0);
    int n, q;
    cin >> n >> q;
    for (int i = 1; i <= n; i++) cin >> a[i];
    build(1, 1, n);

    while (q--) {
        mx = -INF, mi = INF;
        int l, r;
        cin >> l >> r;
        query(1, l, r);
        printf("%d\n", mx - mi);
    }
    return 0;
}
posted @ 2022-04-24 12:21  糖豆爸爸  阅读(88)  评论(0编辑  收藏  举报
Live2D