转载自:最长上升子序列(LIS)长度的O(nlogn)算法

最长上升子序列nlogn算法

在川大oj上遇到一道题无法用n^2过于是,各种纠结,最后习得nlogn的算法

最长递增子序列,Longest Increasing Subsequence 下面我们简记为 LIS。
排序+LCS算法 以及 DP算法就忽略了,这两个太容易理解了。

假设存在一个序列d[1..9] = 2 1 5 3 6 4 8 9 7,可以看出来它的LIS长度为5。n
下面一步一步试着找出它。
我们定义一个序列B,然后令 i = 1 to 9 逐个考察这个序列。
此外,我们用一个变量Len来记录现在最长算到多少了

首先,把d[1]有序地放到B里,令B[1] = 2,就是说当只有1一个数字2的时候,长度为1的LIS的最小末尾是2。这时Len=1

然后,把d[2]有序地放到B里,令B[1] = 1,就是说长度为1的LIS的最小末尾是1,d[1]=2已经没用了,很容易理解吧。这时Len=1

接着,d[3] = 5,d[3]>B[1],所以令B[1+1]=B[2]=d[3]=5,就是说长度为2的LIS的最小末尾是5,很容易理解吧。这时候B[1..2] = 1, 5,Len=2

再来,d[4] = 3,它正好加在1,5之间,放在1的位置显然不合适,因为1小于3,长度为1的LIS最小末尾应该是1,这样很容易推知,长度为2的LIS最小末尾是3,于是可以把5淘汰掉,这时候B[1..2] = 1, 3,Len = 2

继续,d[5] = 6,它在3后面,因为B[2] = 3, 而6在3后面,于是很容易可以推知B[3] = 6, 这时B[1..3] = 1, 3, 6,还是很容易理解吧? Len = 3 了噢。

第6个, d[6] = 4,你看它在3和6之间,于是我们就可以把6替换掉,得到B[3] = 4。B[1..3] = 1, 3, 4, Len继续等于3

第7个, d[7] = 8,它很大,比4大,嗯。于是B[4] = 8。Len变成4了

第8个, d[8] = 9,得到B[5] = 9,嗯。Len继续增大,到5了。

最后一个, d[9] = 7,它在B[3] = 4和B[4] = 8之间,所以我们知道,最新的B[4] =7,B[1..5] = 1, 3, 4, 7, 9,Len = 5。

于是我们知道了LIS的长度为5。

!!!!! 注意。这个1,3,4,7,9不是LIS,它只是存储的对应长度LIS的最小末尾。有了这个末尾,我们就可以一个一个地插入数据。虽然最后一个d[9] = 7更新进去对于这组数据没有什么意义,但是如果后面再出现两个数字 8 和 9,那么就可以把8更新到d[5], 9更新到d[6],得出LIS的长度为6。

然后应该发现一件事情了:在B中插入数据是有序的,而且是进行替换而不需要挪动——也就是说,我们可以使用二分查找,将每一个数字的插入时间优化到O(logN)~~~~~于是算法的时间复杂度就降低到了O(NlogN)~!

#include<cstdio>
#include<iostream>
#include<cstring>
#define MAXN 40005
using namespace std;
int arr[MAXN],ans[MAXN],len;
int binary_search(int i){
int left,right,mid;
left=1,right=len+1;
while(left<=right){
mid = (right+left)>>1;
if(ans[mid]>=arr[i]) right=mid-1;
else left=mid+1;
}
return left;
} int main()
{ int T,p,i,j,k;
scanf("%d",&T);
while(T--){
scanf("%d",&p);
for(i=1; i<=p; ++i)
scanf("%d",&arr[i]);
ans[1] = arr[1];
len=1;
for(i=2; i<=p; ++i){
if(arr[i]>ans[len])
ans[++len]=arr[i];
else{
int pos=binary_search(i);//可以使用lower_bound(ans+1,ans+len+1,arr[i])-ans
ans[pos] = arr[i];
}
}
printf("%d\n",len);
}
return 0;
}

  

最新文章

  1. Javascript模块化开发,使用模块化脚本加载工具RequireJS,提高你代码的速度和质量。
  2. 【转载】Visaul Studio 常用快捷键的动画演示
  3. c# datetime 格式化
  4. 20145317彭垚 《Java程序设计》第8周学习总结
  5. laravel创建新model数据的两种方法
  6. Nginx出现“413 Request Entity Too Large”错误解决方法
  7. java 中能否使用 动态加载的类(Class.forName) 来做类型转换?
  8. JAVA多态需要注意的一些问题
  9. HDU 3328 Flipper (stack)
  10. Java的Exception和Error面试题10问10答
  11. C语言中strcpy,strcmp,strlen,strcat函数原型
  12. VIM编辑器用法
  13. Redis Index
  14. Windows浏览器无法连接VM虚拟机Centos并打开nginx页面
  15. 从Excel获取请求体
  16. 人人开源框架使用 renren fast
  17. svn 剪短笔记
  18. 基于pandas python的美团某商家的评论销售数据分析(可视化)
  19. node启动时候报错 Error: Cannot find module &#39;express&#39;
  20. IE, Firefox下,checkbox的钩钩一旦勾上,画面再刷新,钩钩还是勾上的解决方案

热门文章

  1. 【java多线程】java的线程池分析
  2. test20181020 B君的第二题
  3. Python菜鸟之路:Django 路由、模板、Model(ORM)
  4. windows2008 安装oracle10g“程序异常终止。发生内部错误。请将以下文件提供给oracle技术支持部门
  5. jeecg中的树形控件demo
  6. Windows应用程序的VC链接器设置
  7. python环境搭建,开发环境
  8. Bootstrap-Plugin:附加导航(Affix)插件
  9. RPM包下载网址
  10. django-引用静态文件