zoukankan      html  css  js  c++  java
  • BZOJ4278 : [ONTAK2015]Tasowanie

    首先在串的末尾加上1000,然后进行归并,每次取字典序较小的那个后缀即可。

    用hash+二分支持查询lcp,时间复杂度$O(nlog n)$。

    #include<cstdio>
    typedef long long ll;
    const int N=200010,P=31,D=1000173169;
    int n,m,i,j,k,pow[N],a[N],b[N],f[N],g[N];
    inline void read(int&a){char c;while(!(((c=getchar())>='0')&&(c<='9')));a=c-'0';while(((c=getchar())>='0')&&(c<='9'))(a*=10)+=c-'0';}
    inline int hash0(int l,int r){return(ll)(f[r]-(ll)f[l-1]*pow[r-l+1]%D+D)%D;}
    inline int hash1(int l,int r){return(ll)(g[r]-(ll)g[l-1]*pow[r-l+1]%D+D)%D;}
    inline int ask(int a,int b){
      int l=1,r=n-a+1,mid,t=0;
      if(m-b+1<r)r=m-b+1;
      while(l<=r){
        mid=(l+r)>>1;
        if(hash0(a,a+mid-1)==hash1(b,b+mid-1))l=(t=mid)+1;else r=mid-1;
      }
      return t;
    }
    int main(){
      for(pow[0]=i=1;i<N;i++)pow[i]=(ll)pow[i-1]*P%D;
      for(read(n),i=1;i<=n;i++)read(a[i]),f[i]=((ll)f[i-1]*P+a[i])%D;
      for(read(m),i=1;i<=m;i++)read(b[i]),g[i]=((ll)g[i-1]*P+b[i])%D;
      for(a[n+1]=b[m+1]=1000,i=j=1;i<=n&&j<=m;printf("%d ",a[i+k]<b[j+k]?a[i++]:b[j++]))k=ask(i,j);
      while(i<=n)printf("%d ",a[i++]);
      while(j<=m)printf("%d ",b[j++]);
      return 0;
    }
    

      

  • 相关阅读:
    12迭代器
    11(2)Vector(向量)
    11(1) LinkList ---链表
    11集合(Collection<E>) Arraylist
    10异常
    乘法计算过程的模拟
    10 Date详解
    详细的OA系统学习
    8 math类
    Java开发中的23种设计模式详解
  • 原文地址:https://www.cnblogs.com/clrs97/p/4849424.html
Copyright © 2011-2022 走看看