最长递增子序列

桃扇骨 2022-04-16 06:55 336阅读 0赞

最长递增子序列问题的求解

最长递增子序列问题是一个很基本、较常见的小问题,但这个问题的求解方法却并不那么显而易见,需要较深入的思考和较好的算法素养才能得出良好的算法。由于这个问题能运用学过的基本的算法分析和设计的方法与思想,能够锻炼设计较复杂算法的思维,我对这个问题进行了较深入的分析思考,得出了几种复杂度不同算法,并给出了分析和证明。

一, 最长递增子序列问题的描述

设L=是n个不同的实数的序列,L的递增子序列是这样一个子序列Lin=,其中k1<k2<…<km且aK1<ak2<…<akm。求最大的m值。

二, 第一种算法:转化为LCS问题求解

设序列X=是对序列L=按递增排好序的序列。那么显然X与L的最长公共子序列即为L的最长递增子序列。这样就把求最长递增子序列的问题转化为求最长公共子序列问题LCS了。

最长公共子序列问题用动态规划的算法可解。设Li=< a1,a2,…,ai>,Xj=< b1,b2,…,bj>,它们分别为L和X的子序列。令C[i,j]为Li与Xj的最长公共子序列的长度。则有如下的递推方程:

这可以用时间复杂度为O(n2)的算法求解,由于这个算法上课时讲过,所以具体代码在此略去。求最长递增子序列的算法时间复杂度由排序所用的O(nlogn)的时间加上求LCS的O(n2)的时间,算法的最坏时间复杂度为O(nlogn)+O(n2)=O(n2)。

三, 第二种算法:动态规划法

设f(i)表示L中以ai为末元素的最长递增子序列的长度。则有如下的递推方程:

这个递推方程的意思是,在求以ai为末元素的最长递增子序列时,找到所有序号在L前面且小于ai的元素aj,即j<i且aj<ai。如果这样的元素存在,那么对所有aj,都有一个以aj为末元素的最长递增子序列的长度f(j),把其中最大的f(j)选出来,那么f(i)就等于最大的f(j)加上1,即以ai为末元素的最长递增子序列,等于以使f(j)最大的那个aj为末元素的递增子序列最末再加上ai;如果这样的元素不存在,那么ai自身构成一个长度为1的以ai为末元素的递增子序列。

四, 对第二种算法的改进

在第二种算法中,在计算每一个f(i)时,都要找出最大的f(j)(j<i)来,由于f(j)没有顺序,只能顺序查找满足aj<ai最大的f(j),如果能将让f(j)有序,就可以使用二分查找,这样算法的时间复杂度就可能降到O(nlogn)。于是想到用一个数组B来存储“子序列的”最大递增子序列的最末元素,即有

B[f(j)] = aj

在计算f(i)时,在数组B中用二分查找法找到满足j<i且B[f(j)]=aj<ai的最大的j,并将B[f[j]+1]置为ai。

代码如下:

  1. #include <iostream>
  2. #define LEN 6
  3. using namespace std;
  4. void show(int a[],int N)
  5. {
  6. for(int i=0; i<N; i++) cout<<a[i]<<" ";
  7. cout<<endl;
  8. }
  9. void findMaxLen(int a[],int f[],int n,int len)
  10. {
  11. int i,max=0;
  12. if(n>0)
  13. {
  14. for(i=n-1;i>=0;i--)
  15. {
  16. if(a[i]<=a[n]&&f[i]>=max) max =f[i];
  17. }
  18. }
  19. else{
  20. f[n] = 1;
  21. }
  22. if(max!=0){
  23. f[n] = max + 1;
  24. }
  25. else f[n]=1;
  26. }
  27. int main()
  28. {
  29. int a[LEN]={1,3,4,2,7,5};
  30. int f[LEN]={0};
  31. int max =0;
  32. /* 没有改进的O(n2)
  33. for(int i=0;i<LEN;i++)
  34. {
  35. findMaxLen(a,f,i,5);
  36. if(f[i]>=f[max])max =i;
  37. }
  38. cout<<a[max]<<endl;
  39. show(f,LEN);
  40. */
  41. //改进后的O(nlogn)
  42. int b[LEN]={0};
  43. b[0] = -10000;
  44. b[1] = a[0];
  45. int maxLen=1;
  46. for(int i=1;i<LEN;i++)
  47. {
  48. int mid,left=0,right=maxLen;
  49. while(left<=right)
  50. {
  51. mid = (left+right)/2;
  52. if(b[mid]<a[i])left = mid+1;
  53. else right = mid-1;
  54. }
  55. b[left] = a[i];
  56. if(left > maxLen) maxLen++;
  57. }
  58. cout<<maxLen<<endl;
  59. show(b,LEN);
  60. }

发表评论

表情:
评论列表 (有 0 条评论,336人围观)

还没有评论,来说两句吧...

相关阅读

    相关 递增序列

    问题 给定一个长度为N的数组,找出一个最长的单调自增子序列(不一定连续,但是顺序不能乱)。例如:给定一个长度为6的数组A\{5, 6, 7, 1, 2, 8\},则其...

    相关 递增序列

    给出长度为N的数组,找出这个数组的最长递增子序列。(递增子序列是指,子序列的元素是递增的) 例如:5 1 6 8 2 4 5 10,最长递增子序列是1 2 4 5 10。

    相关 递增序列

    最长递增子序列问题的求解   最长递增子序列问题是一个很基本、较常见的小问题,但这个问题的求解方法却并不那么显而易见,需要较深入的思考和较好的算法素养才能得出良好的算法。由