[BZOJ 3796]Mushroom追妹纸

题目

Mushroom最近看上了一个漂亮妹纸。他选择一种非常经典的手段来表达自己的心意——写情书。考虑到自己的表达能力,Mushroom决定不手写情书。他从网上找到了两篇极佳的情书,打算选择其中共同的部分。另外,Mushroom还有个一个情敌Ertanis,此人也写了封情书给妹子。
Mushroom不希望自己的情书中完整的出现了情敌的情书。(这样抄袭的事情就暴露了)。
Mushroom把两封情书分别用字符串s1和s2来表示,Ertanis的情书用字符串s3来表示,他要截取的部分用字符串w表示。
需满足:
1、w是s1的子串
2、w是s2的子串
3、s3不是w的子串
4、w的长度应尽可能大
所谓子串是指:在字符串中连续的一段

INPUT

输入文件为girl.in
输入有三行,第一行为一个字符串s1第二行为一个字符串s2, 
第三行为一个字符串s3。输入仅含小写字母,字符中间不含空格。

OUTPUT

输出文件为girl.out
输出仅有一行,为w的最大可能长度,如w不存在,则输出0。

SAMPLE

INPUT

abcdef
abcf
bc

OUTPUT

2

解题报告

我不会说我一开始看错题废了我一个来小时的= =

首先我们可以把$s1$和$s2$中间加一个不会出现在该字符集中的字符将它俩接在一起,这个字符日常作用是分隔俩字符串

然后,我们对组合得到的字符串与$s3$跑$KMP$处理出每个匹配的串的位置

接下来就是极其简单的最长公共字串问题了,只要在判断时加一个是否被$KMP$标记过的条件即可

一定要读对题啊啊啊

s3不是w的子串!!!

 #include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
char ch[][];
int len[],bl[];
char ran[]={'!','@','#','$'};
char s[];
int n,m;
int t1[],t2[],t3[],buc[];
int sa[],rank[],height[];
int kp[];
bool km[];
inline void get_kp(){
kp[]=-;
int i(),j(-);
while(i<len[]){
while(j!=-&&ch[][i]!=ch[][j])
j=kp[j];
kp[++i]=++j;
}
}
inline void kmp(){
int i(),j();
while(i<=n){
while(j!=-&&s[i]!=ch[][j])
j=kp[j];
++i,++j;
if(j==len[])
km[i-len[]]=;
}
}
inline void Suffix(){
int i,j,k(),p(),*x(t1),*y(t2),*t;
for(i=;i<=m;++i)buc[i]=;
for(i=;i<=n;++i)++buc[x[i]=s[i]];
for(i=;i<=m;++i)buc[i]+=buc[i-];
for(i=n;i>=;--i)sa[buc[x[i]]--]=i;
for(j=;p<n;j<<=,m=p){
for(p=,i=n-j+;i<=n;++i)y[++p]=i;
for(i=;i<=n;++i)
if(sa[i]>j)
y[++p]=sa[i]-j;
for(i=;i<=m;++i)buc[i]=;
for(i=;i<=n;++i)t3[i]=x[y[i]];
for(i=;i<=n;++i)++buc[t3[i]];
for(i=;i<=m;++i)buc[i]+=buc[i-];
for(i=n;i>=;--i)sa[buc[t3[i]]--]=y[i];
for(t=x,x=y,y=t,x[sa[]]=,p=,i=;i<=n;++i)
x[sa[i]]=((y[sa[i]]==y[sa[i-]])&&(y[sa[i]+j]==y[sa[i-]+j]))?p:++p;
}
for(i=;i<=n;++i)rank[sa[i]]=i;
for(i=;i<=n;height[rank[i++]]=k)
for(k?k--:,j=sa[rank[i]-];s[i+k]==s[j+k];++k);
}
inline bool judge(int po,int x){//cout<<l<<' '<<r<<' '<<x<<endl;
if(x<len[])
return true;
for(int i=po,j=;j<=x;++i,++j){
if(km[i]&&j+len[]-<=x)
return false;
}
return true;
}
inline bool check(int x){//cout<<x<<endl;
int i,j,k;
bool flag[];
for(i=;i<=n;++i){//cout<<i<<" "<<height[i]<<endl;
if(height[i]>=x){//cout<<endl;
for(j=i;height[j]>=x&&j<=n;++j);
--j;
// if(i==j)
// continue;
// cout<<i<<" "<<j<<endl<<endl;
memset(flag,,sizeof(flag));
for(k=i-;k<=j;++k)
flag[bl[sa[k]]]=;
if(flag[]&&flag[]&&judge(sa[i-],x))
return true;
// for(k=1;flag[k]&&k<=3;++k);
// if(k==3)
// return true;
i=j;
}
}
return false;
}
int main(){
scanf("%s%s%s",ch[],ch[],ch[]);
for(int i=;i<=;++i)
len[i]=strlen(ch[i]);
for(int i=;i<=;++i){
for(int j=;j<len[i];++j){
s[++n]=ch[i][j];
bl[n]=i;//cout<<n<<' '<<bl[n]<<endl;
}
s[++n]=ran[i];
}
get_kp();
kmp();
m=;
Suffix();
int l(),r(max(len[],len[])),ans();
while(l<=r){
int mid((l+r)>>);
if(check(mid))
ans=mid,l=mid+;
else
r=mid-;
}
printf("%d",ans);
}

[BZOJ 3796]Mushroom追妹纸的更多相关文章

  1. ●BZOJ 3796 Mushroom追妹纸

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=3796 题解: 题意:    给出三个串 A,B,C    找出一个最长串 S,    使得 ...

  2. bzoj 3796 Mushroom追妹纸——后缀数组

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3796 长度一般都是 1e5 ,看这个是 5e4 ,一看就是把两个串接起来做. 自己本来想的是 ...

  3. bzoj 3796 Mushroom追妹纸 —— 后缀数组

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3796 先把三个串拼在一起,KMP 求 s1 , s2 中每个位置和 s3 的匹配情况: 注意 ...

  4. bzoj 3796: Mushroom追妹纸 AC自动机+后缀自动机+dp

    题目大意: 给定三个字符串s1,s2,s3,求一个字符串w满足: w是s1的子串 w是s2的子串 s3不是w的子串 w的长度应尽可能大 题解: 首先我们可以用AC自动机找出s3在s1,s2中出现的位置 ...

  5. BZOJ 3796 Mushroom追妹纸 哈希+二分(+KMP)

    先把两个串能匹配模式串的位置找出来,然后标记为$1$(标记在开头或末尾都行),然后对标记数组求一个前缀和,这样可以快速查到区间内是否有完整的一个模式串. 然后二分子串(答案)的长度,每次把长度为$md ...

  6. bzoj 3796: Mushroom追妹纸【二分+后缀数组+st表】

    把三个串加上ASCII大于z的分隔符连起来,然后求SA 显然每个相同子串都是一个后缀的前缀,所以枚举s1的每个后缀的最长和s2相同的前缀串(直接在排序后的数组里挨个找,最近的两个分别属于s1和s2的后 ...

  7. 【BZOJ3796】Mushroom追妹纸 二分+hash

    [BZOJ3796]Mushroom追妹纸 Description Mushroom最近看上了一个漂亮妹纸.他选择一种非常经典的手段来表达自己的心意——写情书.考虑到自己的表达能力,Mushroom决 ...

  8. 【bzoj3796】Mushroom追妹纸 hash/sa+kmp+二分

    Description Mushroom最近看上了一个漂亮妹纸.他选择一种非常经典的手段来表达自己的心意--写情书.考虑到自己的表达能力,Mushroom决定不手写情书.他从网上找到了两篇极佳的情书, ...

  9. BZOJ3796 : Mushroom追妹纸

    将S1与S2用#号拼接在一起形成S串 将S3与S串跑KMP求出S3在S串中每次出现的位置l[i] 对于S串每个后缀i,求出f[i]表示该串不包含S3串的最长前缀 然后求出S串的后缀数组 先从小到大扫描 ...

随机推荐

  1. markdownpad2下载安装教程

    1.下载安装 http://markdownpad.com/download/markdownpad2-setup.exe 直接下载,安装过程中提醒要安装微软的一个什么环境,不用理会直接跳过,实测没有 ...

  2. Java初学者如何排查学习中遇到的问题

    大多数新手或者刚入门的人在学习的时候,不管是看视频还是看书,都会遇到各种各样的问题,比如JDK配置了,但是javac没有办法执行,Eclipse安装了,但是打不开,快捷键用不了,照着视频敲了但是和视频 ...

  3. linux如何更改yum源

    更改linux yum源方法:第一步:进入yum配置文件目录:cd /etc/yum.repos.d/第二步:备份配置文件(如果后续出现了问题就可以恢复):mv CentOS-Base.repo Ce ...

  4. swift单例创建的几种方法

    //单例方法1 class SingleTonOne{ static var sharedInstanceOne:SingleTonOne{ struct SingleTonStruct { stat ...

  5. [ NOIP 2009 ] TG

    \(\\\) \(\#A\) \(Spy\) 给出两个长度均为\(N\)相同的样例串,建立第一个串各个字符向第二个串对应位置字符的映射,并用映射转换给出的长度为\(M\)第三个串,输入保证只有大写字符 ...

  6. CSS——background综合运用

    搜索栏图标: <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> < ...

  7. 从CSDN转到cnblogs了

    之前一直用的CSDN的博客,网站卡慢经常出问题,发布文章要审核,连修改几个标点符号也要审核,这些我都忍了,毕竟之前发的文章舍不得弃掉. 现在竟然无故封禁我的博客?请问我写的都是技术文章,有哪点违反规定 ...

  8. DOS批处理命令-字符串操作

    1.字符串替换 语法结构:%变量名:替换前=替换后% @set str=teh cat in teh hat @echo %str% @set str=%str:teh=the% @echo %str ...

  9. hibernate工作流程、session

    hibernate是对jdbc的封装,不建议直接使用jdbc的connection操作数据库,而是通过session操作数据库.session可以理解为操作数据库的对象. session与connec ...

  10. c++选择文件夹对话框

    1,目的 提供一个对话框供用户选择一个文件夹路径. 2,原理&实现 先贴上这个工具类的源码: 在你的程序中使用静态方法 CSelectFolderDlg::Show() 就能显示一个选择文件夹 ...