zoukankan      html  css  js  c++  java
  • LIS最长上升子序列O(n^2)与O(nlogn)的算法

    动态规划

    最长上升子序列问题(LIS)。给定n个整数,按从左到右的顺序选出尽量多的整数,组成一个上升子序列(子序列可以理解为:删除0个或多个数,其他数的顺序不变)。例如序列1, 6, 2, 3, 7, 5,可以选出上升子序列1, 2, 3, 5,也可以选出1, 6, 7,但前者更长。选出的上升子序列中相邻元素不能相等。

    最容易想到的办法就是用一个数组f[i]保存到达第i个数的LIS

    初始化f[i]=1

    更新 f[i]=max{f[j]+1,f[i]|a[j]<a[i],1<=j<i}

    即在第i位置前的比i小的最大的LIS+1

    时间复杂度O(n^2)

    #include<cstdio>
    #include<iostream>//vj1098
    #define ll long long
    #define _max(a,b) ((a)>(b)?(a):(b))
    using namespace std;
    const int N=105;
    int n,a[N],ans;
    int f[N],g[N];
    int main()
    {
        freopen("sample.in","r",stdin);
        cin>>n;
        for(int i=1;i<=n;i++)
        scanf("%d",&a[i]),f[i]=g[i]=1;
        for(int i=1;i<=n;i++)
         for(int j=1;j<i;j++)
          if(a[j]<a[i])
           f[i]=_max(f[i],f[j]+1);
        for(int i=n;i>=1;i--)
         for(int j=n;j>i;j--)
          if(a[j]<a[i])
           g[i]=_max(g[i],g[j]+1);
        for(int i=1;i<=n;i++)
        ans=_max(ans,f[i]+g[i]-1);
        cout<<n-ans;
        return 0;
    }

    从蓝书和网上学到了一种更高效的O(nlogn)的算法

    大概思路如下

      d[i]表示以i结尾的最长的LIS的长度,则d[i]=max{0,d[j]|j<i,Aj<Ai}+1,最终答案是max{d[i]}。如果LIS中的元素可以相等,把小于号改成小于等于号即可。

      假如已经计算出两个状态a,b满足Aa<Ab,且d[a]=d[b],则对于后续所有状态i(即i>a且i>b)来说,a并不会比b差——如果b满足Ab<Aa的条件,a也满足。换句话说,如果我们只保留a,一定不会丢失最优解。

      这样,对于相同的d值,最需要保留A最小的一个。我们用g[i]表示d值为i的最小状态编号(如果不存在,g[i]定义为正无穷)。根据上推理可证明

      g[1]<=g[2]<=g[3]<=……<=g[n]

    #include<cstdio>
    #include<iostream>
    #define ll long long
    #define _max(a,b) ((a)>(b)?(a):(b))
    using namespace std;
    const int N=300005;
    int n,k,a[N],b[N],o[N],ans,ma,mb;
    int j,da[N],db[N],len,la,lb,mid;
    int findpos(int *d,int l,int r,int key){
        while(l<=r){
            mid=(l+r)>>1;
            if(key>d[mid]){
                if(key<=d[mid+1])
                    return mid;
                else l=mid+1;
            }else r=mid-1;
        }return 0;
    }
    int main(){
        cin>>n>>k;
        for(int i=1;i<=n;i++)    scanf("%d",o+i);
        for(int i=1;i<k;i++)    o[i]<o[k]?a[++la]=o[i]:la=la;
        for(int i=k+1;i<=n;i++)    o[i]>o[k]?b[++lb]=o[i]:lb=lb;
        da[1]=a[1],len=1,j=0;
        for(int i=2;i<=la;i++)da[a[i]>da[len]?++len:findpos(da,1,len,a[i])+1]=a[i];
        db[1]=b[1],len=1,j=0;
        for(int i=2;i<=lb;i++)db[b[i]>db[len]?++len:findpos(db,1,len,b[i])+1]=b[i];
        for(int i=la;i>=1;i--)da[i]?ans+=i,i=0:i=i;
        for(int i=lb;i>=1;i--)db[i]?ans+=i,i=0:i=i;
        cout<<ans+1;
        return 0;
    }

    汝佳的code核心

    for(int i=1;i<=n;i++)g[i]=INF;
    for(int i=1;i<=n;i++){
        int k=lower_bound(g+1,g+n+1,A[i])-g;
        d[i]=k;
        g[k]=a[i];
    }
  • 相关阅读:
    文件处理
    集合 字符编码
    3-11作业
    win 10 在vs2017下对mpi的安装以及认识
    java中二维数组的排序
    java中Arrays的用法
    java中随机二维数组中寻找最大值并输出坐标
    用java打印图形
    面向对象object与constructor
    for each in for in 与for of
  • 原文地址:https://www.cnblogs.com/lwhinlearning/p/6052491.html
Copyright © 2011-2022 走看看