数组的最长递减子序列java_最长递增/递减子序列

本文详细解析了编程中寻找数组最长递增子序列的问题,通过比较不同解法,如直接遍历、动态规划、排序优化等,探讨了如何高效地求解最长递增子序列。还介绍了将问题与最长公共子序列问题的关联,以及如何通过排序和数据结构优化算法复杂度。
摘要由CSDN通过智能技术生成

《编程之美》里有个题目是要求数组中最长递增子序列,在CSDN上看到的题目是数组中的最长递减子序列。题目如下:

求一个数组的最长递减子序列

比如{9,4,3,2,5,4,3,2}的最长递减子序列为{9,5,4,3,2}

求一个数组的最长递增子序列

比如{1,-1,2,-3,4,-5,6,-7}的最长递减子序列为{1,2,4,3,6}

最长递增序列和最长递减子序列的解法是一样的,最不济,也可以先revert,求完再revert一次。

首先我们得搞清一个范围问题,虽然我们要的是一个最大值,但是是不是他是可以递推的呢,就是我只保存一个最大值,然后不断增加这个值,直觉上看应该是不太现实的,比如:12345-101234,我们数到5时最大值是5,但往后看时,都比5小,不能增加最大值,但事实上是最大值是6,如果我们受限于前面求的的一个最大值,就会产生错误的结果,也许你可以说,我就是保存一个值,没到一个点我就重新算新的最大值,虽然牺牲一点计算,但是我就想只有一个变量,有思路吗?如果没有就换个思路吧。既然递推困难,我们就把各个备选值都列出来,然后找出那个最大的,那哪些是备选值呢?其实以任意元素为终点的递增序列都是合法的candidate。

解法1:

如同上面的分析,考虑一个序列x0, x1,x2,… xn-1, xn。我们想知道以每个元素为终点的最长递增子序列的长度,令temp[i]表示以xi为终点的递增序列的长度,假设已经知道了temp[0]-temp[n-1],那如何求temp[n]呢?如果xi

#include 

using namespace std;

int LIS(int arr[],int n)

{

int *temp = new int[n];//存放当前遍历位置最长序列    for(int i=0;i

{

temp[i]=1;   //初始化默认长度        for(int j=0;j

//当前值 array[i] 跟已经遍历的值比较,//大于已经遍历的值且已知递增序列+1 大于当前值则 更新当前最长递增序列值            if(arr[i]>arr[j]  && temp[j]+1 > temp[i] )

{

temp[i] = temp[j] + 1;

}

}

}

int max=temp[0];

for(int k=0;k

if(max

max=temp[k];

}

return max;

}

int main()

{

int arr[]={1,-1,2,-3,4,-5,6,-7};

int result=LIS(arr,8);

cout<

}

解法2:

解法1中用一个数组存储以各个元素为终点的最长递增子序列的长度,然后每新增加一个元素时,就遍历前面所有的元素,来知道新元素的temp值,复杂度是O(n2)。每个元素都得求一遍,这是毋庸置疑的,求法上能不能优化一下呢?那就把前面元素和其最长递增序列值排个序吧,当求xn的最长递增序列时,找到前面的序列中比xn小的最大值(可能有多个,位置不同时,序列长度也会不一样),然后加一个1就可以了。代码如下:

解法3:

相类似于解法2,我们不是把序列排序,而是按照序列长度排序,并记录特定长度序列下终点元素的最小值,然后用xn跟这些最小值比,当然我们可以用折半查找,代码如下:

#include 

using namespace std;

int LIS(int array[],int n)

{

int temp[n];//存放当前遍历位置最长序列    int  MaxV[n]; //最长子序列中最大值之间的最小值    MaxV[1]=array[0];//初始序列长度为1的子序列 中最大值的最小值    MaxV[0]=-9999;//边界值

for(int i=0;i

{

temp[i]=1;   //初始化 默认长度    }

int  nMaxLis=1;

int  j;

for(int i=0;i

{

for(j=nMaxLis;j>=0;--j) //找出前面最长的序列        {

if(array[i]>MaxV[j])//当前值大于长度为j的子序列中最大值之间的最小值            {

temp[i] = j + 1;

break;

}

}

if(temp[i]>nMaxLis)//在最长子序列时停止 (这时只有一个最长的)        {

cout<

nMaxLis=temp[i];

MaxV[temp[i]]=array[i];

}

else if(MaxV[j] 

{

MaxV[j+1]=array[i];

}

}

return nMaxLis;

}

int main()

{

int arr[]={1,-1,2,-3,4,-5,6,-7};

int result=LIS(arr,8);

cout<

}

解法4:

还记得我们关于序列的一个经典问题:最长公共子序列。是不是可以转化呢?将原序列排序,并和原序列求最长公共子序列,求得的最长公共子序列长度就是最长递增子序列的长度。

Reference

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值