[模板] LIS
树状数组优化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是不影响最大值的。
所以,我们的大体思路已经成型了:
求出b[1],b[2],…,b[v-1]的最大值
用这个最大值+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的更多相关文章
- [模板]LIS(最长上升子序列)
转载自:最长上升子序列(LIS)长度的O(nlogn)算法 最长上升子序列nlogn算法 在川大oj上遇到一道题无法用n^2过于是,各种纠结,最后习得nlogn的算法 最长递增子序列,Longest ...
- 【小小复习·大米饼】
(一)数位DP模板 ·LIS的数位DP: ·含b进制数个数+数形结合的数位DP ·平衡数的数位DP: (二)网络流问题 ·Edmonds_Karp:(见书)·Dinic(见书)·ISAP(见书)·例题 ...
- 学大伟业 Day 4 培训总结
今天讲的全是dp... 不多废话,先看一道经典的模板LIS(最长不下降子序列) 一.LIS 给定一个长度为N的数列,求最长上升子序列 例:1 7 2 8 3 4 答案:1 2 3 4 代码: #inc ...
- nlogn LIS模板
nlogn 模板 最长上升 #include<bits/stdc++.h> using namespace std; ; int n,x,y,a[N],num[N],d[N],len; / ...
- 动态规划模板1|LIS最长上升子序列
LIS最长上升子序列 dp[i]保存的是当前到下标为止的最长上升子序列的长度. 模板代码: int dp[MAX_N], a[MAX_N], n; int ans = 0; // 保存最大值 for ...
- LIS严格递增和非递减模板
2017-09-10 16:51:03 writer:pprp 严格递增的LIS模板 #include<stdio.h> #include<string.h> #include ...
- LCS/LIS/LCIS 模板总结
/************************* LCS/LIS/LCIs模板总结: *************************/ /*************************** ...
- LIS n^2&nlogn模板
LIS nlogn模板 http://acm.hdu.edu.cn/showproblem.php?pid=1950 #include <iostream> #include <st ...
- 最长上升子序列(LIS)nlogn模板
参考https://www.cnblogs.com/yuelian/p/8745807.html 注意最长上升子序列用lower_bound,最长不下降子序列用upper_bound 比如123458 ...
随机推荐
- 模板 - 动态规划 - 概率期望dp
https://blog.csdn.net/myjs999/article/details/81022546
- loj#2540. 「PKUWC2018」随机算法
传送门 完了pkuwc咋全是dp怕是要爆零了-- 设\(f(S)\)表示\(S\)的排列数,\(S\)为不能再选的点集(也就是选到独立集里的点和与他们相邻的点),\(mx(S)\)表示\(S\)状态下 ...
- 分层图初探 By cellur925
因为最近测试遇到了分层图的题目,所以稍微学了一下==. 这种题目一般是来解决最短路边权有变化/有k条免费路的问题的.他们基本都一般有两种实现方式:dp+最短路/分层图+最短路 当然你如果非要说他们是一 ...
- SQL中进行转列的几种方式
SQL中进行转列 在很多笔试的程序员中会有很多写SQL的情况,其中很多时候会考察行转列.那么这个时候如果能写出来几种行转列的SQL,会给面试官留下比较好的印象. 以下是这次sql转换的表结构以及数据 ...
- oracle 查询用户权限
查询用户和权限 select object_name,created from user_objects; 受权 grant select any table,create table,create ...
- nginx缓存过期
1 原理 在默认下,请求过的内容会接受304,而从本地缓存调用.这是通过client向server发送请求,给出ETag,server确认ETag未变,则不返回内容,client调用本地缓存. 而ex ...
- Hdu 3966 Aragorn's Story (树链剖分 + 线段树区间更新)
题目链接: Hdu 3966 Aragorn's Story 题目描述: 给出一个树,每个节点都有一个权值,有三种操作: 1:( I, i, j, x ) 从i到j的路径上经过的节点全部都加上x: 2 ...
- Sublime3注册码和安装中文包
1.Sublime3注册码 在工具栏Help中点击Enter license,粘贴下面一大串 —– BEGIN LICENSE —– Michael Barnes Single User Licens ...
- MySQL系列:隐式类型转化可能带来的坑
在开发规范中,我们往往会要求研发避免在where条件中出现隐式类型转换,这么要求大概有以下两方面的原因: 隐式类型转换可能导致索引失效: 隐式类型转换可能产生非预期的结果. 注:这里说的是隐式类型转换 ...
- Java基础50题test3—水仙花数
水仙花数 题目:打印出所有的"水仙花数",所谓"水仙花数"是指一个三位数,其各位数字立方和等于该数本身.例 如:153 是一个"水仙花数", ...