<题目链接>

题目大意:

给你一段序列,对其进行两种操作,一是修改某个序号的点的值;二是查询某个区间的LCIS(最长上升子序列)。

解题分析:

线段树区间合并的典型例题,用求某个区间的LCIS时,需要比较三个值,一是左区间的LCIS,二是右区间的LCIS,三是左右子区间合并的LCIS。最重要的是第三点如何实现,实现第三点需要维护一个最长后缀上升子序列和最长前缀上升子序列,总之,相对于一般的线段树,区间合并需要对Pushup()函数进行一些改动,query()的时候也要记得对三种情况进行讨论。

 #include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int siz=;
int a[siz],sum[siz<<],lsum[siz<<],rsum[siz<<]; void pushup(int rt,int l,int r) { //区间合并,更新该节点对应的最长上升子序列和最长前、后缀
lsum[rt]=lsum[rt<<]; //最长前缀
rsum[rt]=rsum[rt<<|]; //最长后缀
sum[rt]=max(sum[rt<<],sum[rt<<|]);
/*-- 先更新该节点需要维护的三个值 --*/ int m=(l+r)>>;
if(a[m]<a[m+]){ //如果左右子区间的LCIS能够合并,那么就进一步更新这三个点 if(lsum[rt<<]==(m-l+)) //如果左子区间的最长前缀==左子区间长度
lsum[rt]+=lsum[rt<<|]; //那么该节点的最长前缀除左子区间的最长前缀以外,还要加上右子区间的最长前缀
if(rsum[rt<<|]==(r-m)) //如果右子区间的最长后缀==右子区间长度
rsum[rt]+=rsum[rt<<]; //那么该节点的最长后缀除右子区间的最长后缀外,还要加上左子区间的最长后缀
/*-- 更新rt节点对应区间的最长前、后缀 --*/ sum[rt]=max(sum[rt],rsum[rt<<]+lsum[rt<<|]);
//更新rt节点对应区间的最长上升子序列
}
}
void build(int l,int r,int rt){
if(l==r){ //注意这里的初始化
sum[rt]=;
lsum[rt]=rsum[rt]=;
return;
}
int m=(l+r)>>;
build(l,m,rt<<);
build(m+,r,rt<<|);
pushup(rt,l,r);
}
void update(int loc,int val,int l,int r,int rt){ //单点更新
if(l==r){
a[l]=val;
return;
}
int m=(l+r)>>;
if(loc<=m)
update(loc,val,l,m,rt<<);
else
update(loc,val,m+,r,rt<<|);
pushup(rt,l,r);
}
int query(int L,int R,int l,int r,int rt){
if(L<=l&&r<=R)
return sum[rt];
int m=(l+r)>>;
int ans=; //注意这里ans初始化为1 /*-- 答案一共三种可能,在左区间或右区间或横跨两个区间 --*/
if(L<=m) //左、右区间的LCIS
ans=max(ans,query(L,R,l,m,rt<<));
if(R>m)
ans=max(ans,query(L,R,m+,r,rt<<|)); if(L<=m&&R>=m&&a[m]<a[m+]) //横跨左右两个子区间的情况
ans=max(ans,min(m-L+,rsum[rt<<])+min(R-m,lsum[rt<<|])); //rsum代表区间最长后缀、lsum为区间最长前缀
return ans;
} int main(){ //lsum为区间左端点开始长度
char c; //rsum为区间右端点开始长度
int t,n,m; //sum为区间最长长度
scanf("%d",&t);
while(t--){
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
scanf("%d",&a[i]);
build(,n,);
while(m--){
int u,v;
cin>>c>>u>>v;
if(c=='Q'){
u++,v++; //因为题目是从0开始编号
printf("%d\n",query(u,v,,n,));
}
else{
u++;
update(u,v,,n,);
}
}
}
return ;
}

2018-09-11

HDU 3308 LCIS (经典区间合并)【线段树】的更多相关文章

  1. BZOJ 2243:染色(树链剖分+区间合并线段树)

    [SDOI2011]染色Description给定一棵有n个节点的无根树和m个操作,操作有2类:1.将节点a到节点b路径上所有点都染成颜色c:2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认 ...

  2. HDU 1540 区间合并线段树

    题目大意: 就是给定一堆位置,进行删除还原,最后找到 t 位置上的最大连续位置 #include <cstdio> #include <cstring> #include &l ...

  3. HDU 3308 LCIS (线段树&#183;单点更新&#183;区间合并)

    题意  给你一个数组  有更新值和查询两种操作  对于每次查询  输出相应区间的最长连续递增子序列的长度 基础的线段树区间合并  线段树维护三个值  相应区间的LCIS长度(lcis)  相应区间以左 ...

  4. HDU 5029 Relief grain(离线+线段树+启发式合并)(2014 ACM/ICPC Asia Regional Guangzhou Online)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5029 Problem Description The soil is cracking up beca ...

  5. hdu 5700区间交(线段树)

    区间交 Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submiss ...

  6. hdu 5274 Dylans loves tree(LCA + 线段树)

    Dylans loves tree Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Othe ...

  7. 【BZOJ4653】【NOI2016】区间(线段树)

    [BZOJ4653][NOI2016]区间(线段树) 题面 BZOJ 题解 \(NOI\)良心送分题?? 既然是最大长度减去最小长度 莫名想到那道反复减边求最小生成树 从而求出最小的比值 所以这题的套 ...

  8. BZOJ_4653_[Noi2016]区间_线段树+离散化+双指针

    BZOJ_4653_[Noi2016]区间_线段树+离散化+双指针 Description 在数轴上有 n个闭区间 [l1,r1],[l2,r2],...,[ln,rn].现在要从中选出 m 个区间, ...

  9. xdoj-1324 (区间离散化-线段树求区间最值)

    思想 : 1 优化:题意是覆盖点,将区间看成 (l,r)转化为( l-1,r) 覆盖区间 2 核心:dp[i]  覆盖从1到i区间的最小花费 dp[a[i].r]=min (dp[k])+a[i]s; ...

随机推荐

  1. WebSocket服务端和客户端使用

    using System;using System.Collections.Generic;using System.IO;using System.Linq;using System.Net;usi ...

  2. OCM 学习练习题目

    1:数据安装操作练习:考试题目 1: Creating a database & Server Configuration --[101]-- #创建数据库 1. Create the dat ...

  3. Confluence 6 数据库表-展现(Appearance)

    这部分存储了有关你 Confluence 的外观和布局使用的信息. decorator 使用自定义 Velocity 布局显示的自定义模板. https://www.cwiki.us/display/ ...

  4. 【Java】SpringBoot配置文件读取中文乱码

    [问题]在配置文件application.properties中配置一个值含有中文的变量.spring加载配置之后,读取的变量中文部分出现乱码.根据CSDN说的一堆办法,改encoding为UTF-8 ...

  5. Pandas模块:表计算与数据分析

    目录 Pandas之Series Pandas之DataFrame 一.pandas简单介绍 1.pandas是一个强大的Python数据分析的工具包.2.pandas是基于NumPy构建的. 3.p ...

  6. jsp 修饰 Request 及Response

    Servlet API包含4个可修饰的类,用于改变Servlet Request以及Servlet Response.这种修饰允许修改 ServletRequest以及ServletResponse或 ...

  7. Python集合(set)

    Python中的集合同数学中的集合概念类似,也是用于保存不重复的元素.他有可变集合(set),和不可变集合(frozenset);可变集合(set)是无序的可变的. 创建集合 直接使用{}创建 set ...

  8. .tar.xz文件的解压方法

    废话不多说: 直接看 方法一: tar -xvJf ***.tar.gz 方法二: 先减压成 .tar 格式的文件, 再解压 .tar #xz是一个工具, 系统中没有安装,需要下载 xz -d *** ...

  9. vue-cli watch简单用法

    创建一个vue单文件 <template> <div id="test"> <h4 @click="changeMsg()" id ...

  10. Pycharm常用操作方法

    1.调整字体大小 2.选择python编译器