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

#include <iostream>
#include <algorithm>
#include <string.h>
#include <algorithm>
#include <vector>
#include <cstdio>
using namespace std;
const int maxn=;
const int maxm=;
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=; }
void build(int m)
{
int *x=t,*y=t2,i;
for(i=; i<m; i++)c[i]=;
for(i=; i<n; i++)c[ x[i] = s[i] ]++;
for(i=; i<m; i++)c[ i ] += c[ i- ];
for(i = n-; i >= ; i--) sa[--c[x[i]]] = i;
for(int k=; k<=n; k<<=)
{
int p=;
for(i=n-k; i<n; i++)y[p++]=i;
for(i=; i<n; i++)if(sa[i]>=k)y[p++]=sa[i]-k;
for(i=; i<m; i++)c[i]=;
for(i=; i<n; i++)c[ x[y[ i] ] ]++;
for(i=; i<m; i++)c[i]+=c[i-];
for(i =n-; i>=; i--)sa[ --c[ x[ y[i] ] ] ] = y[i];
swap(x,y);
p=;
x[ sa[] ] =;
for( i=; i<n; i++) x[ sa[i] ]= y[ sa[i] ]==y[sa[i-]] && y[ sa[i]+k ]== y[ sa[i-] +k ]?p-:p++;
if(p>=n)break;
m=p;
}
}
void build_height() {
int i, k = ;
for(i = ; i < n; i++) rank[sa[i]] = i;
height[]=;
for(i = ; i < n; i++) {
if(k) k--;
if(rank[i]==)continue;
int j = sa[rank[i]-];
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[];
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/) return false;
int cnt = ;
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/;
} void print_sub(int L, int R)
{
for(int i = L; i < R; i++)
printf("%c", sa.s[i] - + 'a');
printf("\n");
}
int ss;
bool print_solutions(int len, bool print)
{
int L = ;
ss++;
for(int R = ; 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(, false))
printf("?\n");
else {
int L = , R = maxlen, M;
while(L < R) {
M = L + (R-L+)/;
if(print_solutions(M, false)) L = M;
else R = M-;
}
print_solutions(L, true);
}
} int main()
{
int kase=;
while(scanf("%d",&n)==)
{
if(n==)break;
if(kase++>) puts("");
maxlen=;
sa.clear();
for(int i=; i<n; i++)
{
scanf("%s",word);
int sz= strlen(word);
maxlen=max(sz,maxlen);
for(int j=; j<sz; j++)
{
add(word[j]-'a'+,i);
}
add(+i,n);
}
add(+n,n);
if(n==)printf("%s\n",word);
else
{
ss=;
memset(flag,,sizeof(flag));
sa.build( +n+ );
sa.build_height();
solve(maxlen);
}
}
return ;
}

uva11107 后缀数组的更多相关文章

  1. UVA11107 Life Forms --- 后缀数组

    UVA11107 Life Forms 题目描述: 求出出现在一半以上的字符串内的最长字符串. 数据范围: \(\sum len(string) <= 10^{5}\) 非常坑的题目. 思路非常 ...

  2. uva11107(后缀数组)

    uva11107 题意 输入 n 个 DNA 序列,求出长度最大的字符串,使得它在超过一半的 DNA 序列中连续出现.如果有多解,按字典序输出. 分析 论文 后缀数组经典题.加深几个关键数组的印象. ...

  3. 【UVA11107 训练指南】Life Forms【后缀数组】

    题意 输入n(n<=100)个字符串,每个字符串长度<=1000,你的任务是找出一个最长的字符串使得超过一半的字符串都包含这个字符串. 分析 训练指南上后缀数组的一道例题,据说很经典(估计 ...

  4. 后缀数组的倍增算法(Prefix Doubling)

    后缀数组的倍增算法(Prefix Doubling) 文本内容除特殊注明外,均在知识共享署名-非商业性使用-相同方式共享 3.0协议下提供,附加条款亦可能应用. 最近在自学习BWT算法(Burrows ...

  5. BZOJ 4199: [Noi2015]品酒大会 [后缀数组 带权并查集]

    4199: [Noi2015]品酒大会 UOJ:http://uoj.ac/problem/131 一年一度的“幻影阁夏日品酒大会”隆重开幕了.大会包含品尝和趣味挑战两个环节,分别向优胜者颁发“首席品 ...

  6. BZOJ 1692: [Usaco2007 Dec]队列变换 [后缀数组 贪心]

    1692: [Usaco2007 Dec]队列变换 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1383  Solved: 582[Submit][St ...

  7. POJ3693 Maximum repetition substring [后缀数组 ST表]

    Maximum repetition substring Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 9458   Acc ...

  8. POJ1743 Musical Theme [后缀数组]

    Musical Theme Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 27539   Accepted: 9290 De ...

  9. 后缀数组(suffix array)详解

    写在前面 在字符串处理当中,后缀树和后缀数组都是非常有力的工具. 其中后缀树大家了解得比较多,关于后缀数组则很少见于国内的资料. 其实后缀数组是后缀树的一个非常精巧的替代品,它比后缀树容易编程实现, ...

随机推荐

  1. probing privatePath如何作用于ASP.NET MVC View

    当View上using一些从probing privatePath加载的程序集,运行时会提示无法找到对应程序集. <runtime> <assemblyBinding xmlns=& ...

  2. windows中安装pip,setuptools,django等

    1,安装Python3.6 (下载exe文件,双击安装)      注意设置环境变量,让Python的在任意位置都可以执行 .Python 下载地址:https://www.python.org/do ...

  3. C# IEqualityComparer 去重

    1.去除list里某重复字段值的数据(相当于group by) public class CorrController { //方法 public void DoGet() { List<tes ...

  4. VS NuGet加载本地程序包

    NuGet是VS中非常实用的一个工具,我们可以通过它在线安装想要的程序包,只要右键点击解决方案中的项目的引用,在弹出的菜单中选择“管理NuGet程序包”,然后就可以通过在线搜索找到想要添加的程序包,下 ...

  5. docker swarn集群笔记

    .安装Docker 三剑客: curl -L https://github.com/docker/machine/releases/download/v0.10.0/docker-machine-`u ...

  6. Spark2 AccumulatorV2累加器

    Accumulator.scala  (Since version 2.0.0) use AccumulatorV2 import org.apache.spark.util._ val accum= ...

  7. Lost connection to MySQL server during query ([Errno 104] Connection reset by peer)

    Can't connect to MySQL server Lost connection to MySQL server during query · Issue #269 · PyMySQL/Py ...

  8. css3的一个小demo(箭头hover变化)

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  9. vue - vue-cli脚手架项目中组件的使用

    在webpack-simple模板中,包括webpck模板.一个.vue文件就是一个组件. 为什么会这样呢?因为webpack干活了!webpack的将我们所有的资源文件进行打包.同时webpack还 ...

  10. webpack学习三——output

    output的两个参数filename,path 一.path输出路径,输出路径要绝对路径,否则报错.做法如下: path:__dirname + 'path' 二.filename 输出文件命,相对 ...