poj 1631 最长上升子序列 nlogn
来源:互联网 发布:新浪邮箱端口号 编辑:程序博客网 时间:2024/05/16 18:44
题目挺长的,就是说 一开始 有一些连线,左边1~n 跟右边 1~n 相连,现在要你 删除一些连线,使得剩下的连线最多,并且剩下的连线不相交。。其实就是求最长上升子序列;
但是 n 的范围 是 40000,n^2 的算法是不行的,需要 nlogn的算法。
算法分析,我们可以维护一个 c数组,这个c数组 保存的是 对长度 为 到目前为止最长上升子序列 最后 一个 值,同时这个值 是最小的,比如 最长 长度 为 3 的时候 1 3 6 和 1 3 5
都是符合条件的,但是 很明显,最后一个 值 c[3]取 5 是较优的。。
那么,可以看出 c 数组 实际上是 单调 递增的,每次我们判断新的一个值的时候,就可以 进行二分查找,查找 c 数组里面 比 当前要加入的数 小的 最后一个数,返回 它的下一个位置 放上 该数,就可以得到一个 尽量长的 序列。。
#include<iostream>#include<algorithm>using namespace std;const int maxn=40005;int a[maxn],c[maxn];int t,n;int bsearch(int size,int val){int low=1,high=size;while(low<=high){int mid=(low+high)>>1;if(val>c[mid]&&val<=c[mid+1])return mid+1;else if(val<c[mid])high=mid-1;else low=mid+1;}}int getLIS(){int i,j;int size=1;c[1]=a[1];for(i=2;i<=n;i++){if(a[i]<=c[1])j=1;else if(a[i]>c[size])j=++size;else j=bsearch(size,a[i]);c[j]=a[i];}return size;}int main(){scanf("%d",&t);while(t--){scanf("%d",&n);for(int i=1;i<=n;i++)scanf("%d",a+i);int ans=getLIS();printf("%d\n",ans);}}
- poj 1631 最长上升子序列 nlogn
- POJ 1631(最长上升子序列 nlogn).
- 最长上升子序列 nlogn poj 2533
- POJ 1631 最长上升子序列O(nlogn)
- 最长上升子序列 nlogn
- nlogn 最长上升子序列
- 最长上升子序列(nlogn)
- Nlogn最长上升子序列
- 最长上升子序列nlogn算法(poj 2533 && SDUT 1299)
- 最长上升子序列nlogn算法(poj 2533 && SDUT 1299)
- 最长上升子序列 nlogn时间复杂度 poj 2533
- poj 1631离散化加树状数组优化 最长上升子序列LIS nlogn做法
- 最长上升子序列nlogn算法
- 最长上升子序列nlogn算法
- 最长上升子序列NLOGN算法
- 最长上升子序列 LIS 〇(nlogn)
- 最长上升子序列O(nlogn)算法
- 最长上升子序列----nlogn算法-模板
- 读书笔记:SQL SERVER 2008高级编程:批处理和脚本
- 如何照顾宝宝
- objective-c 关键字和概念
- Android Launcher开发(五)添加应用程序桌面快捷方常见问题及解决方案
- linux网络高级编程之 fcntl和select使用
- poj 1631 最长上升子序列 nlogn
- Sobel算子边缘检测(vc实现)
- hdu 3714 Error Curves
- 局域网远程桌面使用(基于win7平台)
- 编程需要学习多少数学知识
- linux sar用法
- Linux VIM 搭建自己的IDE 一
- Linux 进程创建及多进程
- jquery 广告轮播