求最长上升子序列长度

动态规划的题目里,这是非常常见的一类题目,因此还是要熟练地掌握才好,我观察了网上的一些解法,觉得有两类解法是值得参考的。
不管是以下哪种方法我们都创建了一个名为A[i]的数组存储数字序列

1.复杂度为o(n^2)

在这个方法中,我们创建了一个名为f[i]的数组来进行动态规划,f[i]存储的数表示以数A[i]为结尾的最长的上升子序列的长度,因此,我们不难得到递推的公式,即每次都判断一下,A[i]是否大于A[j],如果大于,则表示A[i]可以接在f[j]后面,所以此时f[i]=max(f[i],f[j]+1);

#include
#include
#include
using namespace std;
int num[100];
int f[100];
int imax;
int main()
{
 freopen("hehe.txt","r",stdin); 
 int n;
 cin>>n;
 for(int i=1;i<=n;i++)
 cin>>num[i];
 for(int i=1;i<=n;i++)
 {
  for(int j=1;j

2.复杂度为o(n*logn)

这时我们也创建了一个名为f[i]的数组,但是这个数组与第一个解法中的含义是完全不一样的,f[i]存储的数字表示长度为i的上升序列结尾最小的数字。
好的,接下来考虑一个具体的情况
以下是序列 : 1 3 2 5 4 6
首先 f[1]=1;
接着循环 ,轮到3这个数字,考虑到3比1要大,因此f[2]=3;且3无法取代f[1]中1的位置。
接着循环轮到2,因为2比3小而又比1要大因此把f[2]更替为2,f[1]不变。
接着循环轮到5,因为5比2大,因此f[3]=5;并且5无法取代f[1]与f[2]中原本的数字。
接着循环轮到4,因为4比5小而又比2要大因此把f[3]更替为4,f[1]与f[2]不变。
最后轮到了6,因为6比4大,因此f[4]=6循环结束。

发现了吗,我们一直在新的数填进来时想要找俩个数能夹住它,这样它就能更新原本的值。
如果这个值比原本所有值都大,就开创新的f[i]并把它放入。
如果这个值比原本所有值都小,我们就用它替换f[1]。
细心一点就会发现,因为每次我们都是按照大小插入数字或者更新数字,因此f[i]是一个有序的数组,因此我们若是在查找时使用二分查找,就能让我们的复杂度从o(n^2)将为o(nlogn).*
接下来贴上代码:

1.复杂度为o(n^2)的

#include
#include
#include
using namespace std;
int num[100];
int f[100];
int imax;
int main()
{
 freopen("hehe.txt","r",stdin); 
 int n;
 cin>>n;
 for(int i=1;i<=n;i++)
 cin>>num[i];
 for(int i=1;i<=n;i++)
 {
  int j;
  for(j=imax;j>=1;j--)
  {
   if(num[i]>f[j])
   break;
  }
  f[j+1]=num[i];
  if(f[imax+1]!=0)
  imax++;
 }
 cout<

2.复杂度为o(n*logn)的方法

#include
#include
#include
#include
using namespace std;
int num[100];
int f[100];
int imax;
int main()
{
 int * pos; 
 freopen("hehe.txt","r",stdin); 
 int n;
 cin>>n;
 for(int i=1;i<=n;i++)
 cin>>num[i];
 for(int i=1;i<=n;i++)
 {
  pos=lower_bound(f,f+imax,num[i]); //如果没找到就会返回f+imax;c++左闭右开的特性 
  if(pos!=f+imax)
  {
   *pos=num[i];
  }
  else
  f[++imax]=num[i];
 }
 cout<

注意这段代码中用到了lower_bound函数,不熟悉请复习二分查找的应用文章

tip:以上的方法都没有考虑到数为负值的情况。

你可能感兴趣的:(动态规划(ACM))