四合一的题。

简单粗暴的方法:

子串匹配——SAM

子序列匹配——序列自动机

关于序列自动机:序列自动机—— [FJOI2016]所有公共子序列问题

(其实这个玩意没有什么,n+1个点,每个点的字符集的每条出边连向其后的第一个字符,这样保证尽可能用靠前的,后面的能凑出的子序列就能更多,1号点是rt)

类似于SAM,序列自动机的路径条数就是子序列个数。

这样的话,对AB两个串各建造一个SAM和序列自动机

四问就分别在两个机子上bfs跑

如果A有的出边,而B没有,那么返回深度作为答案。

否则A,B都有,入队。

正确性:bfs按深度,会把深度为d的所有公共的子串/子序列搜完后再搜下一个。第一个合法的就是最短的。

复杂度:记忆化一下,因为到了同样一个数对(Ax,By)时候,后面的路径都是固定的了。之前没有找到过,后面也不会找到。直接continue

这样,复杂度就是状态数O(n^2)

就是考察对自动机"路径与子串一一对应"的理解。

#include<bits/stdc++.h>
#define il inline
#define reg register int
#define numb (ch^'0')
using namespace std;
typedef long long ll;
il void rd(int &x){
char ch;bool fl=false;
while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);
(fl==true)&&(x=-x);
}
namespace Miracle{
const int N=;
char a[N],b[N];
int l1,l2;
struct SAM{
int nd,cnt,fa[*N],ch[*N][];
int len[*N];
SAM(){
nd=cnt=;
}
void ins(int c,int pos){
//cout<<c<<" "<<pos<<" "<<cnt<<endl;
int p=nd;nd=++cnt;
len[nd]=pos;
for(;p&&ch[p][c]==;p=fa[p]) ch[p][c]=nd;
if(p==) {fa[nd]=;return;}
int q=ch[p][c];
if(len[q]==len[p]+){fa[nd]=q;return;}
len[++cnt]=len[p]+;
for(reg i=;i<=;++i) ch[cnt][i]=ch[q][i];
fa[cnt]=fa[q];fa[q]=cnt;fa[nd]=cnt;
for(;p&&ch[p][c]==q;p=fa[p]) ch[p][c]=cnt;
}
}SA,SB;
struct XU{
int cnt,ch[N][];
int las[];
void build(char *s,int len){
for(reg i=len;i>=;--i){
for(reg j=;j<=;++j){
if(las[j]) {
ch[i+][j]=las[j];
//cout<<" to "<<i+1<<" "<<j<<" "<<las[j]<<endl;
}
}
if(i)las[s[i]-'a']=i+;
}
}
}XA,XB;
struct po{
int x,y,d;
po(){}
po(int xx,int yy,int dd){
x=xx,y=yy,d=dd;
}
};
queue<po>q;
int vis[*N][*N];
int bfs1(){
while(!q.empty()) q.pop();
q.push(po(,,));
vis[][]=;
while(!q.empty()){
po now=q.front();q.pop();
for(reg i=;i<=;++i){
if(vis[SA.ch[now.x][i]][SB.ch[now.y][i]]==) continue;
vis[SA.ch[now.x][i]][SB.ch[now.y][i]]=;
if(SA.ch[now.x][i]&&!SB.ch[now.y][i]) return now.d+;
if(SA.ch[now.x][i])
{
q.push(po(SA.ch[now.x][i],SB.ch[now.y][i],now.d+));
}
}
}
return -;
}
int bfs2(){
while(!q.empty()) q.pop();
q.push(po(,,));
vis[][]=;
while(!q.empty()){
po now=q.front();q.pop();
for(reg i=;i<=;++i){
if(vis[SA.ch[now.x][i]][XB.ch[now.y][i]]==) continue;
vis[SA.ch[now.x][i]][XB.ch[now.y][i]]=;
if(SA.ch[now.x][i]&&!XB.ch[now.y][i]) return now.d+;
if(SA.ch[now.x][i])
{
q.push(po(SA.ch[now.x][i],XB.ch[now.y][i],now.d+));
}
}
}
return -;
}
int bfs3(){
while(!q.empty()) q.pop();
q.push(po(,,));
vis[][]=;
while(!q.empty()){
po now=q.front();q.pop();
for(reg i=;i<=;++i){
if(vis[XA.ch[now.x][i]][SB.ch[now.y][i]]==) continue;
vis[XA.ch[now.x][i]][SB.ch[now.y][i]]=;
if(XA.ch[now.x][i]&&!SB.ch[now.y][i]) return now.d+;
if(XA.ch[now.x][i])
{
q.push(po(XA.ch[now.x][i],SB.ch[now.y][i],now.d+));
}
}
}
return -;
}
int bfs4(){
while(!q.empty()) q.pop();
q.push(po(,,));
vis[][]=;
while(!q.empty()){
po now=q.front();q.pop();
//cout<<now.x<<" || "<<now.y<<" dd "<<now.d<<endl;
for(reg i=;i<=;++i){
if(vis[XA.ch[now.x][i]][XB.ch[now.y][i]]==) continue;
vis[XA.ch[now.x][i]][XB.ch[now.y][i]]=;
if(XA.ch[now.x][i]&&!XB.ch[now.y][i]) return now.d+;
if(XA.ch[now.x][i])
{
//cout<<" goto "<<XA.ch[now.x][i]<<" "<<
q.push(po(XA.ch[now.x][i],XB.ch[now.y][i],now.d+));
}
}
}
return -;
}
int main(){
scanf("%s",a+);
scanf("%s",b+);
l1=strlen(a+);l2=strlen(b+); for(reg i=;i<=l1;++i) SA.ins(a[i]-'a',i);
for(reg i=;i<=l2;++i) SB.ins(b[i]-'a',i);
//cout<<SA.cnt<<" "<<SB.cnt<<endl;
XA.build(a,l1);XB.build(b,l2);
printf("%d\n%d\n%d\n%d",bfs1(),bfs2(),bfs3(),bfs4());
return ;
} }
signed main(){
Miracle::main();
return ;
} /*
Author: *Miracle*
Date: 2018/12/12 9:10:40
*/

考虑到,我们要最小化一个串的长度,使得这个串在B的机子上跑完回到NULL节点。

所以,也可以DP做。f[i][j]表示,考虑A中前i个位置,匹配到B的机子上的j位置,最小长度。直接在所有机子位置后尝试匹配i+1位,取min即可完成转移。

答案就是f[lenA][NULL]

[HEOI2015]最短不公共子串的更多相关文章

  1. 【BZOJ4032】[HEOI2015]最短不公共子串(后缀自动机,序列自动机)

    [BZOJ4032][HEOI2015]最短不公共子串(后缀自动机,序列自动机) 题面 BZOJ 洛谷 题解 数据范围很小,直接暴力构建后缀自动机和序列自动机,然后直接在两个自动机上进行\(bfs\) ...

  2. BZOJ 4032: [HEOI2015]最短不公共子串

    4032: [HEOI2015]最短不公共子串 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 446  Solved: 224[Submit][Sta ...

  3. 洛谷 P4112 [HEOI2015]最短不公共子串 解题报告

    P4112 [HEOI2015]最短不公共子串 题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的"子串"指的是它的连续的一段,例如bcd是 ...

  4. BZOJ 4032: [HEOI2015]最短不公共子串 后缀自动机 暴力

    4032: [HEOI2015]最短不公共子串 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4032 Description 在虐各种最 ...

  5. bzoj4032: [HEOI2015]最短不公共子串(SAM+DP)

    4032: [HEOI2015]最短不公共子串 题目:传送门 题解: 陈年老题良心%你赛膜爆嘎爷 当初做题...一眼SAM...结果只会两种直接DP的情况... 情况1: 直接设f[i][j] 表示的 ...

  6. luoguP4112 [HEOI2015]最短不公共子串 SAM,序列自动机,广搜BFS

    luoguP4112 [HEOI2015]最短不公共子串 链接 luogu loj 思路 子串可以用后缀自动机,子序列可以用序列自动机. 序列自动机是啥,就是能访问到所有子序列的自动机. 每个点记录下 ...

  7. bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp)

    bzoj4032/luoguP4112 [HEOI2015]最短不公共子串(后缀自动机+序列自动机上dp) bzoj Luogu 题解时间 给两个小写字母串 $ A $ , $ B $ ,请你计算: ...

  8. BZOJ4032[HEOI2015]最短不公共子串——序列自动机+后缀自动机+DP+贪心

    题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...

  9. BZOJ4032:[HEOI2015]最短不公共子串(SAM)

    Description 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列” ...

  10. BZOJ4032: [HEOI2015]最短不公共子串(后缀自动机+序列自动机)

    题目描述 在虐各种最长公共子串.子序列的题虐的不耐烦了之后,你决定反其道而行之. 一个串的“子串”指的是它的连续的一段,例如bcd是abcdef的子串,但bde不是. 一个串的“子序列”指的是它的可以 ...

随机推荐

  1. Delphi方法重载

    unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...

  2. Java学习笔记十一:Java中的方法

    Java中的方法 一:什么是方法: 所谓方法,就是用来解决一类问题的代码的有序组合,是一个功能模块. 学过C语言或者其他语言的应该都知道函数这个东西,在Java中,其实方法就是函数,只不过叫法不同,在 ...

  3. powerpoint教程资料,PPT的

    Powerpoint,是微软公司设计的演示文稿软件,利用Powerpoint不仅可以创建演示文稿,还可以在互联网上召开面对面会议.远程会议或在网上给观众展示演示文稿,掌握利用PowerPoint是一项 ...

  4. mongodb的学习之旅一

    描述 作为一枚菜鸟级别的coder,刚接触nodejs没有多久.现在在学习微信公众号的开发,但是碰到了mongodb保存用户数据的时候,出现了DeprecationWarning: Mongoose: ...

  5. Kubernetes-设计理念(三)

    Kubernetes设计理念与分布式系统 分析和理解Kubernetes的设计理念可以使我们更深入的了解Kubernetes系统,更好的利用它管理分布式部署的云原生应用,另一方面也可以让我们借鉴其在分 ...

  6. VHDL入门学习-程序组成

    1. VHDL程序的组成 一个完整的VHDL程序是以下五部分组成的: 2. 库(LIBRARY):比较好理解,调用系统已有的库,WORK库就是用户当前编辑文件所在的文件夹, IEEE库:由IEEE(美 ...

  7. 步骤1:JMeter 录制脚本接口测试

    JMeter 常用测试方法简介 1.下载安装 http://jmeter.apache.org/download_jmeter.cgi 安装JDK,配置环境变量JAVA_HOME. 系统要求:JMet ...

  8. 系统学习Docker 践行DevOps理念

    Docker代表的容器技术是近两年的大热技术,和人工智能.区块链等热点不同,容器技术的门槛并不高,每一个开发.测试.运维人员都能在日常工作中掌握和使用,是当今IT从业人员的必备技能之一.本课程会带大家 ...

  9. 05-Mysql数据库----补充内容

    数据库命名规则: 数据库命名规则: 可以由字母.数字.下划线.@.#.$ 区分大小写 唯一性 不能使用关键字如 create select 不能单独使用数字 最长128位 # 基本上跟python或者 ...

  10. 信号处理是Unix和LInux系统为了响应某些状况而产生的事件

    信号处理是Unix和LInux系统为了响应某些状况而产生的事件,通常内核产生信号,进程收到信号后采取相应的动作. 例如当我们想强制结束一个程序的时候,我们通常会给它发送一个信号,然后该进程会捕捉到信号 ...