引自:wonter巨巨的博客

定义 dp[i] := 以数字 i(不是下标 i)为结尾的最长上升长度

然后用线段树维护 dp[i]:

每个节点维护 2 个信息,一个是当前区间的最大上升长度,一个是最大上升长度的方案数,





这里再详细说下这篇题解。。。也是弱弱自己的理解吧。。。

以1-n的值为线段树所代表的区间;

然后依次更新,题目就是要找上升序列,那么我们只要每次查询0~arr[i]-1范围内最长的那个长度和方案拿出来;

然后再去0到n区间里更新,更新的值是arr[i],长度为查询到的x+1,方案还是查询到的方案y;

具体查询:

我们是想知道在一段区间(从0到值arr[i]-1,这里一定要注意是值!是值!)的最长长度;

所以一直查询到那个区间,然后一直下去就好了;

中间如果出现被mid是夹在被查询区间中间的话,就要比较一下左右两段区间的最长长度





具体更新:

一直要先更新到最底;

然后再往上更新,

往上更新只要比较一下左右子区间的长度就好了;

#include <bits/stdc++.h>
using namespace std;
const int N=5e5+10;
const int mod=1e9+7; struct asd{
int left,right,mid;
int val;
int cnt;
};
asd q[N*4]; void build(int i,int left,int right)
{
q[i].left=left;
q[i].right=right;
q[i].mid=(left+right)>>1;
q[i].cnt=q[i].val=0;
if(left==right)
return;
build(i*2,left,q[i].mid);
build(i*2+1,q[i].mid+1,right);
} void query(int i, int left, int right, int &x, int &y )
{
if(q[i].left==left&&q[i].right==right)
{
x=q[i].val;
y=q[i].cnt;
return;
}
if(right<=q[i].mid)
{
query(i*2, left, right, x, y);
return;
}
if(left>q[i].mid)
{
query(i*2+1,left,right, x, y);
return;
}
int lx,ly;
int rx,ry;
query(i*2,left,q[i].mid,lx,ly);
query(i*2+1,q[i].mid+1,right, rx,ry);
if(lx==rx)
{
x=lx;
y=(ly+ry)%mod;
}
else if(lx>rx)
{
x=lx;
y=ly;
}
else
{
x=rx;
y=ry;
}
} void update(int i,int p,int x,int y)
{
if(q[i].left==q[i].right&&q[i].left==p)
{
if(x>q[i].val)
{
q[i].val=x;
q[i].cnt=y;
}
else if(x==q[i].val)
{
q[i].cnt=(q[i].cnt+y)%mod;
}
return;
}
if(p<=q[i].mid)
update(i*2,p,x,y);
else if(p>q[i].mid)
update(i*2+1,p,x,y);
if(q[i*2].val==q[i*2+1].val)
{
q[i].val=q[i*2].val;
q[i].cnt=(q[i*2].cnt+q[i*2+1].cnt)%mod;
}
else if(q[i*2].val>q[i*2+1].val)
{
q[i].val=q[i*2].val;
q[i].cnt=q[i*2].cnt;
}
else
{
q[i].val=q[i*2+1].val;
q[i].cnt=q[i*2+1].cnt;
}
} int n;
int arr[N];
vector<int>xs;
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d",&arr[i]);
xs.push_back(arr[i]);
}
sort(xs.begin(),xs.end());
auto e=unique(xs.begin(),xs.end());
for(int i=1;i<=n;i++)
arr[i]=lower_bound(xs.begin(), e, arr[i])-xs.begin()+1; build(1,0,n); for(int i=1;i<=n;i++)
{
int x,y;
query(1,0,arr[i]-1,x,y);
// printf("%d %d\n",x,y);
if(!y)
y=1;
update(1, arr[i] , x+1, y);
}
int x,y;
query(1,0,n,x,y);
printf("%d\n",y);
return 0;
}

51nod 1376【线段树维护区间最大值】的更多相关文章

  1. HDU 2795 Billboard 【线段树维护区间最大值&&查询变形】

    任意门:http://acm.hdu.edu.cn/showproblem.php?pid=2795 Billboard Time Limit: 20000/8000 MS (Java/Others) ...

  2. hdu1754线段树维护区间最大值

    #include <iostream> #include <cstdio> using namespace std; #define MAXN 200005 int N,M; ...

  3. 【HDOJ 1337】I Hate It(线段树维护区间最大值)

    Problem Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少.这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写 ...

  4. 洛谷P3113 [USACO14DEC]马拉松赛跑Marathon_Gold 线段树维护区间最大值 模板

    如此之裸- Code: #include<cstdio> #include<cstring> #include<cmath> #include<algorit ...

  5. POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 )

    POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 ) 题意分析 给出n个点,m个询问,和当前位置pos. 先给出n-1条边,u->v以及边权w. 然后有m个询问 ...

  6. Can you answer these queries V SPOJ - GSS5 (分类讨论+线段树维护区间最大子段和)

    recursion有一个整数序列a[n].现在recursion有m次询问,每次她想知道Max { A[i]+A[i+1]+...+A[j] ; x1 <= i <= y1 , x2 &l ...

  7. 线段树维护区间前k小

    线段树维护区间前k小 $ solution: $ 觉得超级钢琴太麻烦?在这里线段树提供一条龙服务 . 咳咳,开始讲正题!这道题我们有一个和超级钢琴复杂度一样 $ ~O(~\sum x\times lo ...

  8. CodeForces - 587E[线段树+线性基+差分] ->(线段树维护区间合并线性基)

    题意:给你一个数组,有两种操作,一种区间xor一个值,一个是查询区间xor的结果的种类数 做法一:对于一个给定的区间,我们可以通过求解线性基的方式求出结果的种类数,而现在只不过将其放在线树上维护区间线 ...

  9. FJUT3568 中二病也要敲代码(线段树维护区间连续最值)题解

    题意:有一个环,有1~N编号,m次操作,将a位置的值改为b,问你这个环当前最小连续和多少(不能全取也不能不取) 思路:用线段树维护一个区间最值连续和.我们设出两个变量Lmin,Rmin,Mmin表示区 ...

随机推荐

  1. Leetcode 002-Search Insert Position

    #Given a sorted array and a target value, return the index if the target is found. If not, return th ...

  2. Cocos2d-x之LayerMultiplex的使用

    1.用处 用于管理Layer的切换,而不用切换场景. 2.代码 1).h文件 #include "cocos2d.h" #include "ui/CocosGUI.h&q ...

  3. mongodb 安装、启动

    MongoDB 之 你得知道MongoDB是个什么鬼 MongoDB - 1   最近有太多的同学向我提起MongoDB,想要学习MongoDB,还不知道MongoDB到底是什么鬼,或者说,知道是数据 ...

  4. Mall电商项目总结(一)——项目概述

    项目概述 此电商项目为本人学习项目,后端 使用nginx实现负载均衡转发请求到多台tomcat服务器,使用多台 redis服务器分布式 缓存用户登录信息. 项目已经部署到阿里云服务器,从阿里云linu ...

  5. vue axios拦截器介绍

    关于axios的拦截器是一个作用非常大,非常好用的东西.分为请求拦截器和响应拦截器两种.我一般把拦截器写在main.js里. 1. 请求拦截器 请求拦截器的作用是在请求发送前进行一些操作,例如在每个请 ...

  6. servlet理论学习

    servlet是和凭条无关的服务器端的组件,它运行在servlet容器中,servlet容器负责servlet和客户的通信以及调用servlet方法.servlet和客户的通信是采用“请求和响应的模式 ...

  7. linux应用之mysql数据库的安装及配置(centos)

    CentOS下Mysql数据库的安装与配置   如果要在Linux上做j2ee开发,首先得搭建好j2ee的开发环境,包括了jdk.tomcat.eclipse的安装(这个在之前的一篇随笔中已经有详细讲 ...

  8. RightScale发布2017年度云调查报告

    RightScale最近发布了他们的年度云报告(RightScale 2017云现状报告,RightScale 2017 State of the Cloud Report),这份报告包括了云计算在采 ...

  9. HDU2147 kiki's game (SG表找规律)

    Recently kiki has nothing to do. While she is bored, an idea appears in his mind, she just playes th ...

  10. AutoIt中ControlFocus的使用

    在使用AutoIt最控件做自动化操作的时候,经常性的会碰到无法使用Windows Info工具获取控件的属性,但是我们又需要获取该控件的焦点,我们该怎么办呢? 方法1: 应用controlFocus方 ...