【BZOJ 2946】 2946: [Poi2000]公共串 (SAM)
2946: [Poi2000]公共串
Time Limit: 3 Sec Memory Limit: 128 MB
Submit: 1063 Solved: 469Description
给出几个由小写字母构成的单词,求它们最长的公共子串的长度。任务:l 读入单词l 计算最长公共子串的长度l 输出结果Input
文件的第一行是整数 n,1<=n<=5,表示单词的数量。接下来n行每行一个单词,只由小写字母组成,单词的长度至少为1,最大为2000。Output
仅一行,一个整数,最长公共子串的长度。Sample Input
3
abcb
bca
acbcSample Output
2HINT
Source
【分析】
重新学一次SAM,从刷水题开始。
同spoj1812。用第一个串建sam,然后用其他的串跑,ans存在那个点那里,做完一个串的时候还要根据parent边的拓扑序更新该点ans,最后取min即可。
当然后缀数组也是可以的。
- #include<cstdio>
- #include<cstdlib>
- #include<cstring>
- #include<iostream>
- #include<algorithm>
- using namespace std;
- #define Maxn 2010
- int mymax(int x,int y) {return x>y?x:y;}
- int mymin(int x,int y) {return x<y?x:y;}
- struct node
- {
- int pre,son[],step;
- // node() {pre=step=0;memset(son,0,sizeof(son));}
- }t[Maxn*];
- int ans[Maxn*],ad[Maxn*];
- struct sam
- {
- int last,tot;
- /*void upd(int x)
- {
- memset(t[x].son,0,sizeof(t[x].son));
- }*/
- void extend(int k)
- {
- int np=++tot,p=last;
- t[np].step=t[last].step+;
- while(p&&!t[p].son[k])
- {
- t[p].son[k]=np;
- p=t[p].pre;
- }
- if(!p) t[np].pre=;
- else
- {
- int q=t[p].son[k];
- if(t[q].step==t[p].step+) t[np].pre=q;
- else
- {
- int nq=++tot;//upd(tot);
- t[nq].step=t[p].step+;
- memcpy(t[nq].son,t[q].son,sizeof(t[nq].son));
- t[nq].pre=t[q].pre;
- t[q].pre=t[np].pre=nq;
- while(p&&t[p].son[k]==q)
- {
- t[p].son[k]=nq;
- p=t[p].pre;
- }
- }
- }
- last=np;
- }
- }sam;
- char s[],ss[Maxn];
- int main()
- {
- int n;scanf("%d",&n);
- scanf("%s",s);
- sam.last=sam.tot=;
- int l=strlen(s);
- for(int i=;i<l;i++) sam.extend(s[i]-'a'+);
- memset(ans,,sizeof(ans));
- for(int i=;i<=n;i++)
- {
- scanf("%s",s);
- l=strlen(s);
- int nw=,sp=;
- for(int j=;j<=sam.tot;j++) ad[j]=;
- for(int j=;j<l;j++)
- {
- int ind=s[j]-'a'+;
- while(nw&&!t[nw].son[ind]) nw=t[nw].pre,sp=t[nw].step;
- if(t[nw].son[ind]) sp++,nw=t[nw].son[ind];
- else nw=,sp=;
- ad[nw]=mymax(ad[nw],sp);
- }
- for(int i=sam.tot;i>=;i--) ad[t[i].pre]=mymax(ad[t[i].pre],mymin(ad[i],t[t[i].pre].step));
- for(int i=sam.tot;i>=;i--) ans[i]=mymin(ans[i],ad[i]);
- }
- int mx=;
- for(int i=;i<=sam.tot;i++) if(ans[i]<=sam.tot) mx=mymax(mx,ans[i]);
- printf("%d\n",mx);
- return ;
- }
2017-04-17 10:21:42
【BZOJ 2946】 2946: [Poi2000]公共串 (SAM)的更多相关文章
- BZOJ 2946: [Poi2000]公共串
2946: [Poi2000]公共串 Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 787 Solved: 342[Submit][Status][D ...
- BZOJ 2946: [Poi2000]公共串( 后缀自动机 )
一个串建后缀自动机, 其他串在上面跑, 然后用当前串跑的去更新全部 ------------------------------------------------------------------ ...
- 【bzoj2946】[Poi2000]公共串 后缀自动机
[Poi2000]公共串 Time Limit: 3 Sec Memory Limit: 128 MBSubmit: 1386 Solved: 620[Submit][Status][Discus ...
- BZOJ_2946_[Poi2000]公共串_后缀数组+二分答案
BZOJ_2946_[Poi2000]公共串_后缀数组+二分答案 Description 给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 任务: l 读入单 ...
- 【BZOJ2946】[Poi2000]公共串 后缀数组+二分
[BZOJ2946][Poi2000]公共串 Description 给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 任务: l 读入单词 l 计 ...
- [POI2000] 公共串 - 后缀数组,二分
[POI2000] 公共串 Description 给出几个由小写字母构成的单词,求它们最长的公共子串的长度. Solution 预处理出后缀数组和高度数组,二分答案 \(k\) ,对于每一个连续的 ...
- BZOJ 2946 [Poi2000]公共串 (二分+Hash/二分+后缀数组/后缀自动机)
求多串的最长公共字串. 法1: 二分长度+hash 传送门 法2: 二分+后缀数组 传送门 法3: 后缀自动机 拿第一个串建自动机,然后用其他串在上面匹配.每次求出SAM上每个节点的最长匹配长度后,再 ...
- 【BZOJ2946】公共串 [SAM]
公共串 Time Limit: 3 Sec Memory Limit: 128 MB[Submit][Status][Discuss] Description 给出几个由小写字母构成的单词,求它们最 ...
- [BZOJ2946] [Poi2000]公共串解题报告|后缀数组
给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 单词个数<=5,每个单词长度<=2000 尽管最近在学的是SAM...但是看到这个题还是忍不住想写SA... (其实是不 ...
随机推荐
- 使用CSS3创建文字颜色渐变(CSS3 Text Gradient)
考虑一下,如何在网页中达到类似以下文字渐变的效果? 传统的实现中,是用一副透明渐变的图片覆盖在文字上.具体实现方式可参考 http://www.qianduan.net/css-gradient-te ...
- 简单高效的asp.net目录树源代码
前台页面: <%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Default. ...
- css3中新增的样式使用方法
在PC版开发中由于IE原因,我们很少用到css3,但随着平板和智能手机进入我们的生活,以及现在越来越流行,在手机版和平板版开发中我们就可以大胆的使用了,下面我们探讨常用几个css3属性: 1.css3 ...
- [php]http响应头解析
(Status-Line) HTTP/ OK Cache-Control no-cache Content-Length Content-Type image/gif Date Sat, Dec :: ...
- 【CODEVS】1922 骑士共存问题
[算法]二分图最大匹配(最大流) [题解]按(i+j)奇偶性染色后,发现棋子跳到的地方刚好异色. 然后就是二分图了,对于每个奇点向可以跳到的地方连边,偶点不需连(可逆). 所以题目要求转换为求二分图上 ...
- 【BZOJ】4555: [Tjoi2016&Heoi2016]求和 排列组合+多项式求逆 或 斯特林数+NTT
[题意]给定n,求Σi=0~nΣj=1~i s(i,j)*2^j*j!,n<=10^5. [算法]生成函数+排列组合+多项式求逆 [题解]参考: [BZOJ4555][Tjoi2016& ...
- spring-boot支持websocket
spring-boot本身对websocket提供了很好的支持,可以直接原生支持sockjs和stomp协议.百度搜了一些中文文档,虽然也能实现websocket,但是并没有直接使用spring-bo ...
- [洛谷P1029]最大公约数与最小公倍数问题 题解(辗转相除法求GCD)
[洛谷P1029]最大公约数与最小公倍数问题 Description 输入二个正整数x0,y0(2<=x0<100000,2<=y0<=1000000),求出满足下列条件的P, ...
- 在Unity中实现屏幕空间反射Screen Space Reflection(1)
本篇文章我会介绍一下我自己在Unity中实现的SSR效果 出发点是理解SSR效果的原理,因此最终效果不是非常完美的(代码都是够用就行),但是从学习的角度来说足以学习到SSR中的核心算法. 如果对核心算 ...
- bzoj 2741 可持久化trie
首先我们设si为前i个数的xor和,那么对于询问区间[i,j]的xor和,就相当于si-1^sj,那么对于这道题的询问我们可以处理处si,然后对于询问[l,r],可以表示为在区间[l-1,r]里找两个 ...