[线段树] HDU 3306

本文介绍了一种使用线段树解决连续递增子序列问题的方法。通过维护节点上的前缀、后缀及最大连续递增子序列长度,实现高效的区间查询与更新操作。文章提供了完整的代码示例,并解释了关键算法细节。

感觉线段树的递归还好hhh

但被这个连续递增子序列折腾的够呛hhh

每个节点不仅仅要保存当前最长序列的大小,还要保存前缀和后缀最长序列的大小,以达到正确合并

(自己完全想不出来hhhh)

开始看了代码也是一阵懵逼,把树画出来之后好看很多




#include <algorithm>
#include <cmath>
#include <cstring>
#include <iostream>

#define lson rt << 1
#define rson rt << 1 | 1

using namespace std;

const int e = 1000000;

int num[ e ];
int pre[ e << 2 ];     //保存当前区间从最左开始,连续增长序列的大小(前缀)
int suf[ e << 2 ];      //保存当前区间从最右开始,连续增长序列的大小(后缀)
int mid[ e << 2 ];      //保存当前区间整个,连续增长序列的大小    

// 修改pre,mid,suf保证区间内的序列长度
void pushup ( int rt, int l, int r ) {
        mid[ rt ] = max ( mid[ lson ], mid[ rson ] );  //默认的序列大小为左右两个孩子的序列大小中较大的数
        pre[ rt ] = pre[ lson ];                       //根的前缀由左孩子的前缀决定
        suf[ rt ] = suf[ rson ];                       //根的后缀由右孩子的后缀决定

        int m = ( l + r ) >> 1;
        int len = r - l + 1;

        if ( num[ m ] < num[ m + 1 ] ) {               //当前根的左右孩子可以连成一条新的连续序列
                mid[ rt ] = max ( mid[ rt ], suf[ lson ] + pre[ rson ] );    //左孩子后缀 + 右孩子前缀

                if ( pre[ lson ] == len - len / 2 )    //如果左孩子前缀的大小等于左孩子这段区间的大小
                        pre[ rt ] += pre[ rson ];      //则根节点的前缀就可以再连上右孩子的前缀
                if ( suf[ rson ] == len / 2 )          //右孩子同理
                        suf[ rt ] += suf[ lson ];

        }
}

void build ( int rt, int l, int r ) {
        if ( l == r ) {
                mid[ rt ] = pre[ rt ] = suf[ rt ] = 1;
                return;
        }

        int m = ( l + r ) >> 1;
        build ( lson, l, m );
        build ( rson, m + 1, r );
        pushup ( rt, l, r );
}

//单节点更新
void update ( int rt, int l, int r, int idx, int val ) {
        if ( l == r )
                if ( l == idx ) {
                        num[ idx ] = val;
                        pre[ rt ] = mid[ rt ] = suf[ rt ] = 1;
                        return;
                }

        int m = ( l + r ) >> 1;
        if ( idx <= m )
                update ( lson, l, m, idx, val );
        if ( idx > m )
                update ( rson, m + 1, r, idx, val );

        pushup ( rt, l, r );
}

// l,r-->now
// L,R-->required
int query ( int rt, int l, int r, int L, int R ) {
        //完全在区间里
        if ( l >= L && r <= R )
                return mid[ rt ];

        int m = ( l + r ) >> 1;
        
        //完全在左孩子的区间里
        if ( R <= m )
                return query ( lson, l, m, L, R );
        //完全在右孩子的区间里
        if ( L > m )
                return query ( rson, m + 1, r, L, R );

        //部分在左区间部分在右区间

        int res = max ( query ( lson, l, m, L, R ), query ( rson, m + 1, r, L, R ) );  
        if ( num[ m ] < num[ m + 1 ] ) {
                int prex = min ( suf[ lson ], m - L + 1 );   //取左孩子后缀 和 所求区间在左孩子区间里的大小 的 较小值
                int suf = min ( pre[ rson ], R - m );
                res = max ( res, prex + suf );
        }

        return res;
}

int main () {
        int T;
        scanf ( "%d", &T );
        while ( T-- ) {
                int n, m;
                scanf ( "%d%d", &n, &m );

                for ( int i = 1; i <= n; ++i )
                        scanf ( "%d", &num[ i ] );

                build ( 1, 1, n );

                while ( m-- ) {
                        char ins[ 5 ];
                        int a, b;
                        scanf ( "%s%d%d", ins, &a, &b );
                        if ( ins[ 0 ] == 'U' )
                                update ( 1, 1, n, ++a, b );
                        else if ( ins[ 0 ] == 'Q' ) {
                                int c = query ( 1, 1, n, ++a, ++b );
                                printf ( "%d\n", c );
                        }
                }
        }

        return 0;
}


乐播投屏是一款简单好用、功能强大的专业投屏软件,支持手机投屏电视、手机投电脑、电脑投电视等多种投屏方式。 多端兼容与跨网投屏:支持手机、平板、电脑等多种设备之间的自由组合投屏,且无需连接 WiFi,通过跨屏技术打破网络限制,扫一扫即可投屏。 广泛的应用支持:支持 10000+APP 投屏,包括综合视频、网盘与浏览器、美韩剧、斗鱼、虎牙等直播平台,还能将央视、湖南卫视等各大卫视的直播内容一键投屏。 高清流畅投屏体验:腾讯独家智能音画调校技术,支持 4K 高清画质、240Hz 超高帧率,低延迟不卡顿,能为用户提供更高清、流畅的视觉享受。 会议办公功能强大:拥有全球唯一的 “超级投屏空间”,扫码即投,无需安装。支持多人共享投屏、远程协作批注,PPT、Excel、视频等文件都能流畅展示,还具备企业级安全加密,保障会议资料不泄露。 多人互动功能:支持多人投屏,邀请好友加入投屏互动,远程也可加入。同时具备一屏多显、语音互动功能,支持多人连麦,实时语音交流。 文件支持全面:支持 PPT、PDF、Word、Excel 等办公文件,以及视频、图片等多种类型文件的投屏,还支持网盘直投,无需下载和转格式。 特色功能丰富:投屏时可同步录制投屏画面,部分版本还支持通过触控屏或电视端外接鼠标反控电脑,以及在投屏过程中用画笔实时标注等功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值