hdu 3308 LCIS(线段树区间合并)

Posted Gealo

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了hdu 3308 LCIS(线段树区间合并)相关的知识,希望对你有一定的参考价值。

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3308

题意:给你n个数,m个操作。操作有两种:1.U x y 将数组第x位变为y   2. Q x y 问数组第x位到第y位连续最长子序列的长度。对于每次询问,输出一个答案

 

题解:一道简单的线段树区间合并,一般线段树的区间合并都会设lsum,rsum,表示左连续和右连续还有sum表示总共的连续。

转移比较复杂那这题为例

int L = T[i << 1].r , R = T[(i << 1) | 1].l;

    T[i].l = T[i << 1].l , T[i].r = T[(i << 1) | 1].r;

    T[i].lsum = T[i << 1].lsum , T[i].rsum = T[(i << 1) | 1].rsum;

    T[i].sum = max(T[i << 1].sum , T[(i << 1) | 1].sum);

    if(a[L] < a[R] && R - L == 1) {

        if(T[i << 1].lsum == T[i << 1].r - T[i << 1].l + 1) {

            T[i].lsum += T[(i << 1) | 1].lsum;

        }//注意看是否可以合并

        if(T[(i << 1) | 1].rsum == T[(i << 1) | 1].r - T[(i << 1) | 1].l + 1) {

            T[i].rsum += T[i << 1].rsum;

        }//注意看是否可以合并

        T[i].sum = max(T[i].sum , T[i << 1].rsum + T[(i << 1) | 1].lsum);

    }

区间合并注意T[i].lsum的更新和T[i].rsum的更新,sum的更新比较简单。

#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
const int M = 1e5 + 10;
int a[M];
struct TnT {
    int l , r , lsum , rsum , sum;
}T[M << 2];
void push_up(int i) {
    int L = T[i << 1].r , R = T[(i << 1) | 1].l;
    T[i].l = T[i << 1].l , T[i].r = T[(i << 1) | 1].r;
    T[i].lsum = T[i << 1].lsum , T[i].rsum = T[(i << 1) | 1].rsum;
    T[i].sum = max(T[i << 1].sum , T[(i << 1) | 1].sum);
    if(a[L] < a[R] && R - L == 1) {
        if(T[i << 1].lsum == T[i << 1].r - T[i << 1].l + 1) {
            T[i].lsum += T[(i << 1) | 1].lsum;
        }
        if(T[(i << 1) | 1].rsum == T[(i << 1) | 1].r - T[(i << 1) | 1].l + 1) {
            T[i].rsum += T[i << 1].rsum;
        }
        T[i].sum = max(T[i].sum , T[i << 1].rsum + T[(i << 1) | 1].lsum);
    }
}
void build(int l , int r , int i) {
    int mid = (l + r) >> 1;
    T[i].l = l , T[i].r = r , T[i].lsum = 1 , T[i].rsum = 1 , T[i].sum = 1;
    if(l == r) return ;
    build(l , mid , i << 1);
    build(mid + 1 , r , (i << 1) | 1);
    push_up(i);
}
void updata(int pos , int ad , int i) {
    int mid = (T[i].l + T[i].r) >> 1;
    if(T[i].l == pos && T[i].r == pos) {
        a[pos] = ad;
        return ;
    }
    if(mid < pos) updata(pos , ad , (i << 1) | 1);
    else updata(pos , ad , i << 1);
    push_up(i);
}
int query(int l , int r , int i) {
    int mid = (T[i].l + T[i].r) >> 1;
    if(T[i].l == l && T[i].r == r) {
        return T[i].sum;
    }
    push_up(i);
    if(mid < l) {
        return query(l , r , (i << 1) | 1);
    }
    else if(mid >= r) {
        return query(l , r , i << 1);
    }
    else {
        int ans = max(query(l , mid , i << 1) , query(mid + 1 , r , (i << 1) | 1));
        ans = max(ans , 1);
        if(a[T[i << 1].r] < a[T[(i << 1) | 1].l] && T[i].l != T[i].r) {
            return max(ans , min(T[i << 1].rsum , mid - l + 1) + min(T[(i << 1) | 1].lsum , r - mid));
        }
        else {
            return ans;
        }
    }
}
int main() {
    int t , n , m , x , y;
    scanf("%d" , &t);
    while(t--) {
        scanf("%d%d" , &n , &m);
        for(int i = 0 ; i < n ; i++) {
            scanf("%d" , &a[i]);
        }
        char cp[10];
        build(0 , n , 1);
        while(m--) {
            scanf("%s" , cp);
            scanf("%d%d" , &x , &y);
            if(cp[0] == ‘U‘) {
                updata(x , y , 1);
            }
            else {
                printf("%d\n" , query(min(x , y) , max(x , y) , 1));
            }
        }
    }
    return 0;
}

以上是关于hdu 3308 LCIS(线段树区间合并)的主要内容,如果未能解决你的问题,请参考以下文章

hdu--3308 LCIS(线段树+区间合并)

hdu 3308 LCIS(线段树区间合并)

hdu 3308 LCIS(线段树)

[HDOJ3308]LCIS(线段树,区间合并,新的代码)

[HDOJ3308]LCIS(线段树,区间合并)

HDU3308(LCIS) 线段树好题