码迷,mamicode.com
首页 > 编程语言 > 详细

uva11107 后缀数组

时间:2015-05-14 15:44:52      阅读:168      评论:0      收藏:0      [点我收藏+]

标签:

题意给了n个串 然后计算 这些串中的子串在大于1/2的串中出现 求出这个串的最长长度。 将这些串用一个每出现的不同的字符拼起来 ,然后二分找lcp

技术分享
#include <iostream>
#include <algorithm>
#include <string.h>
#include <algorithm>
#include <vector>
#include <cstdio>
using namespace std;
const int maxn=2005;
const int maxm=205;
const int MM=maxn*maxm;
struct SuffixArray
{
  int s[MM];      // 原始字符数组(最后一个字符应必须是0,而前面的字符必须非0)
  int sa[MM];     // 后缀数组
  int rank[MM];   // 名次数组. rank[0]一定是n-1,即最后一个字符
  int height[MM]; // height数组
  int t[MM], t2[MM], c[MM]; // 辅助数组
  int n; // 字符个数
     void clear(){ n=0; }
  void build(int m)
  {
      int *x=t,*y=t2,i;
      for(i=0; i<m; i++)c[i]=0;
      for(i=0; i<n; i++)c[ x[i] = s[i] ]++;
      for(i=1; i<m; i++)c[ i ] += c[ i-1 ];
     for(i = n-1; i >= 0; i--) sa[--c[x[i]]] = i;
      for(int k=1; k<=n; k<<=1)
      {
           int p=0;
           for(i=n-k; i<n; i++)y[p++]=i;
           for(i=0; i<n; i++)if(sa[i]>=k)y[p++]=sa[i]-k;
           for(i=0; i<m; i++)c[i]=0;
           for(i=0; i<n; i++)c[  x[y[ i] ] ]++;
           for(i=1; i<m; i++)c[i]+=c[i-1];
           for(i =n-1; i>=0; i--)sa[  --c[ x[ y[i] ]  ]  ] = y[i];
           swap(x,y);
           p=1;
           x[ sa[0] ] =0;
           for( i=1; i<n; i++) x[ sa[i] ]= y[ sa[i] ]==y[sa[i-1]] && y[ sa[i]+k ]== y[ sa[i-1] +k ]?p-1:p++;
           if(p>=n)break;
           m=p;
      }
  }
   void build_height() {
    int i,  k = 0;
    for(i = 0; i < n; i++) rank[sa[i]] = i;
    height[0]=0;
    for(i = 0; i < n; i++) {
      if(k) k--;
      if(rank[i]==0)continue;
      int j = sa[rank[i]-1];
      while(i+k<n&&j+k<n&&s[i+k] == s[j+k]) k++;
      height[rank[i]] = k;
    }
  }
}sa;
char word[MM];
int idx[MM],n,maxlen;
int flag[150];
void add(int ch, int i)
{
  idx[sa.n] = i;
  sa.s[sa.n++] = ch;
}
// 子串[L,R) 是否符合要求
bool good(int L, int R,int &ss)
{

  if(R - L <= n/2) return false;
  int cnt = 0;
  for(int i = L; i < R; i++) {
    int x = idx[sa.sa[i]];
    if(x != n && flag[x]!=ss) { flag[x] = ss; cnt++; }
  }
  return cnt > n/2;
}

void print_sub(int L, int R)
{
  for(int i = L; i < R; i++)
    printf("%c", sa.s[i] - 1 + a);
  printf("\n");
}
int ss;
bool print_solutions(int len, bool print)
{
  int L = 0;
ss++;
  for(int R = 1; R <= sa.n; R++) {
    if(R == sa.n || sa.height[R] < len) { // 新开一段
      if(good(L, R,ss)) {
        if(print) print_sub(sa.sa[L], sa.sa[L] + len); else return true;
      }
      ss++;
      L = R;
    }
  }
  return false;
}

void solve(int maxlen)
{
  if(!print_solutions(1, false))
    printf("?\n");
  else {
    int L = 1, R = maxlen, M;
    while(L < R) {
      M = L + (R-L+1)/2;
      if(print_solutions(M, false)) L = M;
      else R = M-1;
    }
    print_solutions(L, true);
  }
}

int main()
{
    int kase=0;
    while(scanf("%d",&n)==1)
        {
             if(n==0)break;
             if(kase++>0) puts("");
               maxlen=0;
              sa.clear();
              for(int i=0; i<n; i++)
                {
                    scanf("%s",word);
                    int sz= strlen(word);
                    maxlen=max(sz,maxlen);
                    for(int j=0; j<sz; j++)
                        {
                            add(word[j]-a+1,i);
                        }
                    add(123+i,n);
                }
                add(123+n,n);
                if(n==1)printf("%s\n",word);
                else
                    {
                        ss=1;
                        memset(flag,0,sizeof(flag));
                        sa.build( 123+n+4 );
                        sa.build_height();
                        solve(maxlen);
                    }
        }
    return 0;
}
View Code

 

uva11107 后缀数组

标签:

原文地址:http://www.cnblogs.com/Opaser/p/4503350.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!