树状数组优化LIS到nlogn,网上找了好多,感觉讲得都不是很明白,正好自己复习整理一下。

基本的DP方程 f[i]=max(f[i],f[j]+1) (j<i且a[j]<a[i])

定义一个数组b

以a[i]为下标

存储 以a[i]结尾的最长上升子序列长度

例如 序列 5 2 1 3 4

最终b数组如下

i 1 2 3 4 5
b 1 1 2 3 1

定义它有什么好处呢?

每次转移f[i],是找f[j]的过程,有两个条件:

1. f[j]最大值

2. j<i

第二个j<i好说,用循环卡范围在[1,i-1]即可。

那第一个条件呢?a在[1,i-1]的最大值,区间最值,RMQ问题?

用什么呢?ST表?不行,我们要更改。

注意到这个区间一定是一个前缀区间,那树状数组呢? 树状数组本质是求前缀和的,当然可以改造成求前缀最大值啦。

void query(int now){
int ret=0;
while(now){
ret=max(ret,t[now]);
now-=now&-now;
}
return ret;
} void updata(int now,int w){
while(now<=n){
t[now]=max(t[now],w);
now+=now&-now;
}
}

再具体考虑一下转移,先考虑限制a[j]<a[j]的问题

把a数组离散化,1-n个下标对应1-n个数

定义v为a[i]

那就是求出b[1],b[2],… ,b[v-1]的最大值,这个可以用树状数组。

那j<i的问题呢?

换言之,b[1~v-1]是可以求最大值,那怎么*保证这些比a[i]小的数一定在i前面*呢?

答案是:不需要保证

在i前面的数,之前已经更新过,所以b数组中这些数有了值,而i之后的数,尚未访问过,就是0。

0是不影响最大值的。

所以,我们的大体思路已经成型了:

  1. 求出b[1],b[2],…,b[v-1]的最大值

  2. 用这个最大值+1更新b[v]

再次强调上述的v是里当前正在更新的那个数,也就是a[i],i是下标,v是树状数组的下标。

所以,这个b数组不需要真实存在,用树状数组假装它存在,实际上维护前缀最大值就行。

复杂度估计一下,循环遍历n个数,每次找前缀最大值logn,合计O(nlogn)

//Stay foolish,stay hungry,stay young,stay simple
#include<iostream>
#include<algorithm>
using namespace std; const int MAXN=20000; int a[MAXN],tmp[MAXN],f[MAXN];
int n; int t[MAXN];
inline void updata(int now,int w){
while(now<=n){
t[now]=max(t[now],w);
now+=now&-now;
}
}
inline int query(int now){
int ret=0;
while(now){
ret=max(ret,t[now]);
now-=now&-now;
}
return ret;
} int main(){
cin>>n;
for(int i=1;i<=n;i++){
cin>>a[i];
tmp[i]=a[i];
}
sort(tmp+1,tmp+1+n);
int tot=unique(tmp+1,tmp+1+n)-tmp;
for(int i=1;i<=n;i++)
a[i]=lower_bound(tmp+1,tmp+1+tot,a[i])-tmp;
int ans=1;
for(int i=1;i<=n;i++){
f[i]=query(a[i]-1)+1;
updata(a[i],f[i]);
ans=max(f[i],ans);
}
cout<<ans<<endl;
return 0;
}

[模板] LIS的更多相关文章

  1. [模板]LIS(最长上升子序列)

    转载自:最长上升子序列(LIS)长度的O(nlogn)算法 最长上升子序列nlogn算法 在川大oj上遇到一道题无法用n^2过于是,各种纠结,最后习得nlogn的算法 最长递增子序列,Longest ...

  2. 【小小复习·大米饼】

    (一)数位DP模板 ·LIS的数位DP: ·含b进制数个数+数形结合的数位DP ·平衡数的数位DP: (二)网络流问题 ·Edmonds_Karp:(见书)·Dinic(见书)·ISAP(见书)·例题 ...

  3. 学大伟业 Day 4 培训总结

    今天讲的全是dp... 不多废话,先看一道经典的模板LIS(最长不下降子序列) 一.LIS 给定一个长度为N的数列,求最长上升子序列 例:1 7 2 8 3 4 答案:1 2 3 4 代码: #inc ...

  4. nlogn LIS模板

    nlogn 模板 最长上升 #include<bits/stdc++.h> using namespace std; ; int n,x,y,a[N],num[N],d[N],len; / ...

  5. 动态规划模板1|LIS最长上升子序列

    LIS最长上升子序列 dp[i]保存的是当前到下标为止的最长上升子序列的长度. 模板代码: int dp[MAX_N], a[MAX_N], n; int ans = 0; // 保存最大值 for ...

  6. LIS严格递增和非递减模板

    2017-09-10 16:51:03 writer:pprp 严格递增的LIS模板 #include<stdio.h> #include<string.h> #include ...

  7. LCS/LIS/LCIS 模板总结

    /************************* LCS/LIS/LCIs模板总结: *************************/ /*************************** ...

  8. LIS n^2&nlogn模板

    LIS nlogn模板 http://acm.hdu.edu.cn/showproblem.php?pid=1950 #include <iostream> #include <st ...

  9. 最长上升子序列(LIS)nlogn模板

    参考https://www.cnblogs.com/yuelian/p/8745807.html 注意最长上升子序列用lower_bound,最长不下降子序列用upper_bound 比如123458 ...

随机推荐

  1. 51nod 1062【水题】

    直接打表构造啊 #include <stdio.h> #include <string.h> #include <iostream> using namespace ...

  2. 11.6NOIP模拟赛

    [数据规模和限制] 对于全部测试数据,满足 N,M,K≤,W≤ 各个测试点的数据规模及特殊性质如下表. 测试点 N M K ≤ ≤ ≤ ≤ ≤ ≤ ≤ ≤ ≤ ≤ 师 更多咨询:北京信息学窦老师 QQ ...

  3. 洛谷P2515 [HAOI2010]软件安装(tarjan缩点+树形dp)

    传送门 我们可以把每一个$d$看做它的父亲,这样这个东西就构成了一个树形结构 问题是他有可能形成环,所以我们还需要一遍tarjan缩点 缩完点后从0向所有入度为零的点连边 然后再跑一下树形dp就行了 ...

  4. 【POJ - 2376】Cleaning Shifts(贪心)

    Cleaning Shifts Descriptions: 原文是English,我这就直接上Chinese了,想看原文的点一下链接哦 大表哥分配 N (1 <= N <= 25,000) ...

  5. Docker部署Tomcat,Nginx,Redis应用

    一.tomcat部署 1.拉取tomcat镜像 docker pull tomcat:7-jre7 2创建tomcat容器 创建容器用于部署单点登录系统(CAS)  -p表示地址映射 docker r ...

  6. django相关命令

    1 安装django pip3 install django 2 django-admin命令 django-admin startproject mysite #创建一个项目 3 manage.py ...

  7. scikit-learning API

    API参考 这是scikit学习的类和函数参考.有关详细信息,请参阅完整的用户指南,因为类和功能原始规格可能不足以给出其使用的完整指导. sklearn.base:基类和效用函数 所有估计器的基类. ...

  8. Traffic Real Time Query System HDU - 3686

    https://vjudge.net/problem/HDU-3686 点双啊,就是在求割顶的时候,另外用一个栈来存一些边 在遍历u点出发的边时,遇到树边或反向边(u,v)就把此边加入栈(可能要记一下 ...

  9. ACM_X章求和(数学)

    X章求和 Time Limit: 2000/1000ms (Java/Others) Problem Description: X章最喜欢求和了,他一看到什么鬼就什么鬼都加起来.one day,他得到 ...

  10. 题解报告:hdu 2141 Can you find it?(二分)

    Problem Description Give you three sequences of numbers A, B, C, then we give you a number X. Now yo ...