zoukankan      html  css  js  c++  java
  • LIS(最长上升子序列) (n*log2*n)

    li 给定两个长度分别为nm的序列,序列中的每个元素都是正整数。保证每个序列中
    的各个元素互不相同。求这两个序列的最长公共子序列的长度。
     
     
     
    解法2:贪心+二分:
    思路:
    新建一个 low 数组,low [ i ]表示长度为i的LIS结尾元素的最小值。对于一个上升子序列,显然其结尾元素越小,越有利于在后面接其他的元素,也就越可能变得更长。因此,我们只需要维护 low 数组,对于每一个a[ i ],如果a[ i ] > low [当前最长的LIS长度],就把 a [ i ]接到当前最长的LIS后面,即low [++当前最长的LIS长度] = a [ i ]。
    那么,怎么维护 low 数组呢?
    对于每一个a [ i ],如果a [ i ]能接到 LIS 后面,就接上去;否则,就用 a [ i ] 取更新 low 数组。具体方法是,在low数组中找到第一个大于等于a [ i ]的元素low [ j ],用a [ i ]去更新 low [ j ]。如果从头到尾扫一遍 low 数组的话,时间复杂度仍是O(n^2)。我们注意到 low 数组内部一定是单调不降的,所有我们可以二分 low 数组,找出第一个大于等于a[ i ]的元素。二分一次 low 数组的时间复杂度的O(lgn),所以总的时间复杂度是O(nlogn)。
      我们再举一个例子:有以下序列A[ ] = 3 1 2 6 4 5 10 7,求LIS长度。
      我们定义一个B[ i ]来储存可能的排序序列,len 为LIS长度。我们依次把A[ i ]有序地放进B[ i ]里。
         (为了方便,i的范围就从1~n表示第i个数)
      A[1] = 3,把3放进B[1],此时B[1] = 3,此时len = 1,最小末尾是3
      A[2] = 1,因为1比3小,所以可以把B[1]中的3替换为1,此时B[1] = 1,此时len = 1,最小末尾是1
      A[3] = 2,2大于1,就把2放进B[2] = 2,此时B[ ]={1,2},len = 2
      同理,A[4]=6,把6放进B[3] = 6,B[ ]={1,2,6},len = 3
      A[5]=4,4在2和6之间,比6小,可以把B[3]替换为4,B[ ] = {1,2,4},len = 3
      A[6] = 5,B[4] = 5,B[ ] = {1,2,4,5},len = 4
      A[7] = 10,B[5] = 10,B[ ] = {1,2,4,5,10},len = 5
      A[8] = 7,7在5和10之间,比10小,可以把B[5]替换为7,B[ ] = {1,2,4,5,7},len = 5
      最终我们得出LIS长度为5。但是,但是!!这里的1 2 4 5 7很明显并不是正确的最长上升子序列。是的,B序列并不表示最长上升子序列,它只表示相应最长子序列长度的排好序的最小序列。这有什么用呢?我们最后一步7替换10并没有增加最长子序列的长度,而这一步的意义,在于记录最小序列,代表了一种“最可能性”。假如后面还有两个数据8和9,那么B[6]将更新为8,B[7]将更新为9,len就变为7,可以自行体会它的作用。
      因为在B中插入的数据是有序的,不需要移动,只需要替换,所以可以用二分查找插入的位置,那么插入n个数的时间复杂度为〇(logn),这样我们会把这个求LIS长度的算法复杂度降为了〇(nlogn)。话不多说了,show me the code!
    code  map函数可以开够1000000000 的数组
    //
    #include<stdio.h>
    #include<bits/stdc++.h>
    using namespace std;
    int f[400010];
    int n,m;
    int a1[400100],a2[400010];
    map<int , int > pos;
    int c[400010];
    int lis[400010];
    int tot;
    int p;
    int main()
    {
        cin>>n>>m;
        for(int i=1;i<=n;i++)
        {
            scanf("%d",&a1[i]);
            pos[a1[i]]=i;
        }
        for(int i=1;i<=m;i++)
        {
            scanf("%d",&a2[i]);
            if(pos[a2[i]]) c[++tot]=pos[a2[i]];
        }
        lis[1]=c[1];
        int le=1;
        for(int i=2;i<=tot;i++)
        {
            if(c[i]>lis[le]) lis[++le]=c[i];
            else
            {
                if(c[i]<=lis[le])  p=lower_bound(lis+1,lis+1+le,c[i])-lis;
                lis[p]=c[i];
            }
        }
        cout<<le;
        
        
    }
    刀剑映出了战士的心。而我的心,漆黑且残破
  • 相关阅读:
    干货分享:如何使用Kubernetes的Ingress API
    十年OpenStack Ussuri最新版发布 主要改进在可靠性、安全性和用例支持等方面
    如何更好地优化容器的创建?这些技巧你务必收藏
    Kubernetes是容器化微服务的圣杯么?
    微服务是否真的需要服务网格?
    ZOOM火速收购加密公司Kaybase 能否补齐安全短板?
    5个实例告诉您:如何实施成功的容器化多云策略
    新基建火了,开源云计算渠道能做什么?
    盘点6个Kubernetes监视工具
    掌握这10种方法帮你快速在Linux上分析二进制文件
  • 原文地址:https://www.cnblogs.com/OIEREDSION/p/11228044.html
Copyright © 2011-2022 走看看