给出几个由小写字母构成的单词,求它们最长的公共子串的长度。
单词个数<=5,每个单词长度<=2000
  
 
  尽管最近在学的是SAM...但是看到这个题还是忍不住想写SA...
  (其实是不知道应该怎么用SAM做...
  对于后缀数组而言,多个字符串的公共子串与两个处理起来并没有什么区别
  只要在中间加一些没有用的字符,将多个字符串拼成一个字符串
  然后二分答案,对于一个长度L,在一组除了开头其他height都>=L的区间中如果每个字符串的位置都出现过就可以
  
  应该是第二次这么解决一道公共串的题了..
  然后发现了一些新的东西..
  比如之前处理没有的字符串处理了很久,这次我直接在每两个字符串间添加不同的字符
  理由很简单:
  对于原来就在同一个字符串中的两个后缀,它们后面的字符相同,但是由于长度不同,所以没有影响
  即匹配的时候谁先接触到无用字符谁就更小一些,符合我们的愿望
  而对于原来不在一个字符串中的两个后缀,如果他们后面的字符相同,再加上原本的长度也一样
  匹配的时候height数组算出的答案就会比实际的长度要大
  然而我们现在填了不同的字符就可以很好地避免掉这个问题
 
  不知道为什么...我的SA很长...
  不过无所谓只要好理解不用背代码...长一点也只是敲敲键盘的问题...
  这里是用SAM解这道题的方法
 
 
 program bzoj2946;
const maxn = ;
var b,sa,rank,tmp,a,pos,height:array[-..maxn]of longint;
vis,l,r:array[-..]of longint;
maxlen,n,m,i,j,lx,rx,mid,ans:longint;
s:ansistring;
ss:array[-..]of ansistring; function max(a,b:longint):longint;
begin
if a>b then exit(a) else exit(b);
end; procedure Suffix_Array;
var i,j,p,sz,v0,v1,v01,v00:longint;
begin
sz:=max(,length(s));
for i:= to sz do b[i]:=;
for i:= to m- do inc(b[a[i]]);
for i:= to m- do rank[i]:=a[i];
for i:= to sz do inc(b[i],b[i-]);
for i:=m- downto do
begin
dec(b[rank[i]]);
sa[b[rank[i]]]:=i;
end;
j:=;
while j<=m do
begin
p:=;
for i:=m-j to m- do
begin
tmp[p]:=i;inc(p);
end;
for i:= to m- do if sa[i]-j>= then
begin
tmp[p]:=sa[i]-j;inc(p);
end;
for i:= to sz do b[i]:=;
for i:= to m- do inc(b[rank[i]]);
for i:= to sz do inc(b[i],b[i-]);
for i:=m- downto do
begin
dec(b[rank[tmp[i]]]);
sa[b[rank[tmp[i]]]]:=tmp[i];
end;
tmp[sa[]]:=;p:=;
for i:= to m- do
begin
v0:=sa[i-];v1:=sa[i];
if v0+j<m then v00:=rank[v0+j] else v00:=-;
if v1+j<m then v01:=rank[v1+j] else v01:=-;
if (rank[v0]=rank[v1])and(v00=v01) then tmp[sa[i]]:=p else
begin
inc(p);tmp[sa[i]]:=p;
end;
end;
for i:= to m- do rank[i]:=tmp[i];
j:=j << ;
end;
end; function compare(i,j,x:longint):longint;
begin
while (i+x-<m)and(j+x-<m)and(a[i+x-]=a[j+x-]) do inc(x);
exit(x-);
end; procedure calc_height;
var i:longint;
begin
if rank[]= then height[]:= else height[]:=compare(,sa[rank[]-],);
for i:= to m- do
if rank[i]= then height[i]:= else height[i]:=compare(i,sa[rank[i]-],max(height[i-],));
end; function solve(x:longint):boolean;
var i,j,k:longint;
begin
for i:= to n do vis[i]:=-;
i:=;
while i<m do
begin
j:=i+;
while (j<m)and(height[sa[j]]>=x) do inc(j);
for k:=i to j- do vis[pos[sa[k]]]:=i;
for k:= to n do if vis[k]<>i then break;
if vis[k]=i then exit(true);
i:=j;
end;
exit(false);
end; begin
readln(n);s:='';maxlen:=;
for i:= to n do
begin
readln(ss[i]);
if length(ss[i])>maxlen then maxlen:=length(ss[i]);
end;
maxlen:= << (trunc(ln(maxlen)/ln())+);
fillchar(pos,sizeof(pos),);
for i:= to n do
begin
l[i]:=length(s)+;
s:=s+ss[i];
r[i]:=length(s);
for j:=l[i] to r[i] do pos[j-]:=i;
for j:= to maxlen do s:=s+chr(i);
end;
m:=length(s);
for i:= to m- do a[i]:=ord(s[i+]);
Suffix_Array;
Calc_Height;
Lx:=;Rx:=maxlen;ans:=;
while Lx<=Rx do
begin
mid:=(Lx+Rx) >> ;
if solve(mid) then
begin
ans:=mid;Lx:=mid+;
end else Rx:=mid-;
end;
writeln(ans);
end.

[BZOJ2946] [Poi2000]公共串解题报告|后缀数组的更多相关文章

  1. [BZOJ2946][Poi2000]公共串解题报告|后缀自动机

    鉴于SAM要简洁一些...于是又写了一遍这题... 不过很好呢又学到了一些新的东西... 这里是用SA做这道题的方法 首先还是和两个字符串的一样,为第一个字符串建SAM 然后每一个字符串再在这个SAM ...

  2. [bzoj2946][Poi2000]公共串_后缀数组_二分

    公共串 bzoj-2946 Poi-2000 题目大意:给定$n$个字符串,求他们的最长公共子串. 注释:$1\le n\le 5$,$1\le minlen<maxlen\le 2000$. ...

  3. BZOJ2946 Poi2000 公共串 【后缀自动机】

    Description 给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 任务: l 读入单词 l 计算最长公共子串的长度 l 输出结果 Input 文件的第一行是整数 n,1<=n& ...

  4. [codevs3160]最长公共子串解题报告|后缀自动机

    给出两个由小写字母组成的字符串,求它们的最长公共子串的长度. 样例就觉得不能更眼熟啊...好像之前用后缀数组做过一次 然后发现后缀自动机真的好好写啊...(当然当时学后缀数组的时候也这么认为... 这 ...

  5. [BZOJ2754] [SCOI2012]喵星球上的点名解题报告|后缀数组

    a180285幸运地被选做了地球到喵星球的留学生.他发现喵星人在上课前的点名现象非常有趣.   假设课堂上有N个喵星人,每个喵星人的名字由姓和名构成.喵星球上的老师会选择M个串来点名,每次读出一个串的 ...

  6. [BZOJ3238][Ahoi2013]差异解题报告|后缀数组

    Description 先分析一下题目,我们显然可以直接算出sigma(len[Ti]+len[Tj])的值=(n-1)*n*(n+1)/2 接着就要去算这个字符串中所有后缀的两两最长公共前缀总和 首 ...

  7. bzoj2946 [Poi2000]公共串(SA,SAM)

    [题意] 多串求LCS.   [思路]   主要是想找一下SAM的优越感 :) velui good 后缀数组划分height需要注意不少细节 <_<,然后不停debug   [代码]   ...

  8. BZOJ2946 [Poi2000]公共串(后缀自动机)

    Description          给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 任务: l        读入单词 l        计算最长公共子串的长度 l        输 ...

  9. SPOJ1812: LCS2 - Longest Common Substring II & BZOJ2946: [Poi2000]公共串

    [传送门:SPOJ1811&BZOJ2946] 简要题意: 给出若干个字符串,求出这些字符串的最长公共子串 题解: 后缀自动机 这两道题的区别只是在于一道给出了字符串个数,一个没给,不过也差不 ...

随机推荐

  1. 【数据库】 SQL 常用语句之系统语法

    [数据库] SQL 常用语句之系统语法 1. 获取取数据库服务器上所有数据库的名字 SELECT name FROM master.dbo.sysdatabases 2. 获取取数据库服务器上所有非系 ...

  2. Anytime项目开发记录0

    Anytime,中文名:我很忙. 开发者:孤独的猫咪神. 这个项目会持续更新,直到我决定不再维护这个APP. 2014年3月10日:近日有事,暂时断更.希望可以会尽快完事. 2014年3月27日:很抱 ...

  3. dispaly:-webkit-box 布局中的坑

    dispaly:-webkit-box 具体用法 这里大家可以网上查, 这里说下里面的坑 里面的子对象设置-webkit-box-flex: 1 -webkit-box-flex: 2 时:一般两个子 ...

  4. jmeter3.2版本如何进行webservice接口功能测试

    jmeter3.2版本之后就没有SOAP/XML-RPC Request插件了,所以没办法直接进行webservice接口的测试. 原理上: Web service一般就是用SOAP协议通过HTTP来 ...

  5. React获取数据,假如为数组,使用map出现的问题

    在平时做项目的时候,使用到了redux, 如果获取服务器端的数据,例如返回一个  data = [1,2,3,4]data.map(item => item*2) , 这样使用的话如果数据正常获 ...

  6. APP功能性测试-4

    弱网络测试 使用fiddler模拟低速环境 使用fiddler抓取手机上某个应用的包 手机连接fiddler fiddler 代理地址127.0.0.1默认端口8888 只抓http协议(https, ...

  7. NO1——线段树

    /* 数组存储 */ /* 预处理 */ #include <iostream> #include <cstdio> #include <algorithm> #i ...

  8. android桌面悬浮窗仿QQ手机管家加速效果

    主要还是用到了WindowManager对桌面悬浮进行管理. 需要一个火箭的悬浮窗 一个发射台悬浮窗  ,判断火箭是否放到了发射台,如果放上了,则使用AsyTask 慢慢将火箭的图片往上移.结束后., ...

  9. oracle带条件的Insert语句

    背景 在一条记录完结时,自动向表中加入一条新的记录,采用的是事务处理,修改现有记录,并新增一条记录,直接采用的insert语句会报错 //主键冲突 unique constraint (XXXXXX) ...

  10. Android隐藏键盘

    今天接到的任务是在验证码输入框中加入键盘监听事件,需要点击Enter实现登录,这个比较好实现,但是在登录时,键盘并没有隐藏掉,看上去很别扭,因此,百度了一堆方法,但是都无济于事,最后找到了一个,如下, ...