最长上升子序列问题 nlogn 实现算法的简述

首先举个例子说明最长上升子序列(longest increasing subsequence 缩写 LIS):

  1,4,6,2,3,7,5 中1,2,3,5 和1,4,6,7都是最长上升子序列,长度均为4,且相邻元素不能相等。

LIS是动态规划中的经典问题,O(n2)的做法是设d(i)为以i为结尾的最长上升子序列的长度,状态转移方程为:d[i]=max{0,d[j]|j<i,A[j]<A[i]}+1。

下面我们仔细思考以下情况:

  i<j时,d[i]=d[j],显然这种情况只能是A[i]>=A[j];这时我们计算 d[t](t>j且A[t]>A[i]),那么应优先选取以A[j]结尾的子序列作为A[t]的前缀序列,因为如果存在

i<j<z<t,满足A[j]<A[z]<A[i]<A[t],子序列的长度会因z的存在而增加。

  由此我们使用数组D[k]保存满足d[t]=k的最小A[t],即D[k]=min{A[t]|d[t]=k};

  可以证明D[k]是严格单调递增的,即D[1]<D[2]<D[3]<……<D[len],

证明如下:

  D[k]=min{A[t1]|d[t1]=k};

  D[k+1]=min{A[t2]|d[t2]=k+1};

  采用反证法,

  令A[t1]=D[k],A[t2]=D[k+1]

  假设A[t2]<A[t1];

  设以A[t2]结尾对应的子序列为S[1]~S[k],A[t2],  满足S[k]<A[t2].

  显然S[1]~S[k]是一个以S[k]为结尾的最长上升子序列,长度为k,

  则有A[t1]=D[k]<=S[k]<A[t2],与假设矛盾,故D为严格单调递增序列。

于是利用D我们可以得到另外一种计算最长上升子序列的方法,并且可以边读边计算D,算法如下:

  1)设当前最大子序列长度为len,读入A[i];

  2)如果A[i]>D[len],则len++,D[len]=A[i];

  3)如果A[i]<=D[len],则从1~len中二分查找第一个k,使D[k]>=A[i],更新D[k]=A[i].

代码如下:

 1   int n;//原序列长度
 2     cin>>n;
 3     memset(A, 0,sizeof A);
 4     memset(D, 0, sizeof D);
 5     int len=0;//当前最长子序列长度
 6     for(int i=0;i<n;i++){
 7         cin>>A[i];
 8         if(A[i]>D[len]){
 9             len++;
10             D[len]=A[i];
11         }
12         else {
13             int k=lower_bound(D, D+len, A[i])-D;//二分搜索D[k]>=A[i],更新D[k]
14             D[k]=A[i];
15         }
16     }
17     cout<<len<<endl;
18     return 0;

 

posted @ 2016-05-18 20:54  kiraa  阅读(439)  评论(0编辑  收藏  举报