题目意思:

有两个字符串(长度\(<=80\)),按字典序输出它们的最长公共子串的所有情况.

解析

最长公共子序列的长度应该都没问题了吧...有问题请自行百度

但关键是要求出每种情况,还要按字典序...

考虑到长度\(<=80\),我们可以用搜索+剪枝,

设两个串为\(a\),\(b\),长度为\(la\),\(lb\),

\(f[i][j]\)表示\(a\)串中的\(1\)~\(i\)与\(b\)串中的$ 1\(~\)j$的最长公共子序列的长度.

先求出最长公共子序列的长度\(len\)=\(f[la][lb]\),

那么.我们就是要求出一个长度为\(len\)的\(a\),\(b\),的子串,

我们再设\(f1[i][j]\)表示在\(a\)串的\(1\)~\(j\)中,字母\(i\)出现的最后一个位置,

\(f2[i][j]\)就表示\(b\)串,其中\(i\)取\(1\)~\(26\),

那么在搜索时,记录\(4\)个信息:\(a\)串还能取的长度\(l1\),\(b\)串还能取的长度\(l2\),当前求出的子串\(s\)和还需要求的子串的长度\(l\).

那么,在一开始时,状态就应该是(\(la,lb,"",len\)),其中\(s\)是一个空串.

那么,在搜索时枚举每个字母\(ci\),设\(p1\)表示\(f1[ci][l1]\),\(p2\)表示\(f2[ci][l2]\),

那么,若\(f[p1][p2]>=l\),即最终能形成长度为\(len\)的子串,就将\(ci\)加入\(s\),

并且注意,由于我们是倒着搜的,所以要用\(ci+s\)而不是\(s+ci\).

当\(l1\)或\(l2\)小于\(0\)时,即搜不到了,就返回,

而当\(l\)等于\(0\)时就添加答案并返回.

最后排序后输出就行了.

具体实现看代码吧:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <string>
#include <algorithm>
using namespace std; inline int read(){
int sum=0,f=1;char ch=getchar();
while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0' && ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
return f*sum;
} int f[101][101],tot=0;
char a[101],b[101];
int f1[27][101],f2[27][101];
string ans[1001]; void find(int l1,int l2,string s,int l){
if(l1<0||l2<0) return ;
if(l<=0){ans[++tot]=s;return ;}
for(int i=1;i<=26;i++){
int p1=f1[i][l1],p2=f2[i][l2];
if(f[p1][p2]!=l) continue;
char c=i+96;
find(p1-1,p2-1,c+s,l-1);
}
} int main(){
cin>>a>>b;
int la=strlen(a),lb=strlen(b);
for(int i=la;i;i--) a[i]=a[i-1];
for(int i=lb;i;i--) b[i]=b[i-1];//这两步是整体位移,个人习惯而已
for(int i=1;i<=26;i++){
for(int j=1;j<=la;j++){
if(a[j]==i+96) f1[i][j]=j;
else f1[i][j]=f1[i][j-1];
}
for(int j=1;j<=lb;j++){
if(b[j]==i+96) f2[i][j]=j;
else f2[i][j]=f2[i][j-1];
}
}//记录f1,f2
for(int i=1;i<=la;i++){
for(int j=1;j<=lb;j++){
f[i][j]=max(f[i-1][j],f[i][j-1]);
if(a[i]==b[j]) f[i][j]=max(f[i-1][j-1]+1,f[i][j]);
}
}//寻找最长公共子序列
find(la,lb,"",f[la][lb]);
sort(ans+1,ans+tot+1);
for(int i=1;i<=tot;i++) cout<<ans[i]<<endl;
return 0;
}

题解 【POJ1934】 Trip的更多相关文章

  1. 【题解】POJ1934 Trip (DP+记录方案)

    [题解]POJ1934 Trip (DP+记录方案) 题意: 传送门 刚开始我是这么设状态的(谁叫我DP没学好) \(dp(i,j)\)表示钦定选择\(i\)和\(j\)的LCS,然而你会发现这样钦定 ...

  2. POJ1934 Trip 题解

    LCS 模板,但要输出具体方案,这就很毒瘤了. 神奇的预处理:fa[i][j]表示在 \(a\) 串的前 \(i\) 个字符中,字母表第 \(j\) 个字母最晚出现的位置,fb[i][j]同理. 这样 ...

  3. poj1934 Trip【线性DP】【输出方案】

    Trip Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 3850   Accepted: 1030 Description ...

  4. [POJ1934] Trip

    问题描述 Alice and Bob want to go on holiday. Each of them has planned a route, which is a list of citie ...

  5. SPOJ33&POJ1934 Trip LCS

    题目传送门:https://www.luogu.org/problemnew/show/SP33 题目大意:给出两个字符串,求其LCS(最长公共子序列)的长度与具体方案(相同的串算作同一方案).数据组 ...

  6. $Poj1934\ Trip$ 线性$DP+$搜索

    Luogu Description 爱丽丝和鲍伯想去度假,他们每个人都制定了一个参观城市的清单,该地区正好有26个城市,因此它们被编码为小写字母“a”到“z”.清单上可能重复出现某个城市.因为他们想一 ...

  7. 常规DP专题练习

    POJ2279 Mr. Young's Picture Permutations 题意 Language:Default Mr. Young's Picture Permutations Time L ...

  8. [题解] trip

    题目大意 给定一颗大小为 \(N\) 的树, \(1\)的度数不小于 \(2\) .每个点有一个颜色,要么为黑色要么为白色. 从 \(1\) 号点开始游走,计数器初始为 \(0\). 如果当前为黑点计 ...

  9. Codeforces Round #365 (Div. 2) Mishka and trip

    Mishka and trip 题意: 有n个城市,第i个城市与第i+1个城市相连,他们边的权值等于i的美丽度*i+1的美丽度,有k个首都城市,一个首都城市与每个城市都相连,求所有边的权值. 题解: ...

随机推荐

  1. 第一章 Scala基础篇

    目录 一.Scala基础语法 (一) 变量.类型.操作符 1.变量申明 2.字符串 3.数据类型 4.操作符 (二)循环判断 1.块表达式 2.条件表达式 3.循环表达式 (三)方法和函数 1.方法 ...

  2. TCP的粘包、半包和Netty的处理

    参考文献:极客时间傅健老师的<Netty源码剖析与实战>Talk is cheap.show me the code! 什么是粘包和半包 在客户端发送数据时,实际是把数据写入到了TCP发送 ...

  3. 【记录】看见的一些很好的博客x存一下

    [字符串] AC自动机:https://www.cnblogs.com/cjyyb/p/7196308.html

  4. Vanya and Scales CodeForces - 552C (思维)

    大意: $101$个砝码, 重$w^0,w^1,...,w^{100}$, 求能否称出重量$m$. w<=3时显然可以称出所有重量, 否则可以暴力双端搜索. #include <iostr ...

  5. 【sublime Text】sublime Text3安装可以使xml格式化的插件

    应该有机会 ,会碰到需要格式化xml文件的情况. 例如,修改word转化的xml文件之后再将修改之后的xml文件转化为word文件. 但是,word另存的xml文件是没有格式的一片: 那怎么格式化 这 ...

  6. Postgresql在Windows下的解压安装

    1.将下载的压缩包解压,我是解压在D:\postgreSQL\pgsql中. 2.设置环境变量如下: set PGHOME=D:\postgreSQL\pgsql    set PGDATA=%PGH ...

  7. java——多线程知识点大总结

    1:理解线程的概念之前,我们有必要先理解一下进程的概念 程序(Program)是为实现特定目标或解决特定问题而用计算机语言(比如Java语言)编写的命令序列的集合. 进程指一个程序的一次执行过程   ...

  8. vs code 快捷键设置:选中字母可以快速全部转换为大写或小写

    文件--->首选项--->键盘快捷方式--->搜索:"大写"--->点击"转换为大写"左侧的加号,然后设置快捷键后按enter即可完成添 ...

  9. MySQL的基本操作一

    本文主要涉及到的SQL知识点包括CREATE创建数据库和表.INSERT插入数据.SUM()求和.GROUP BY分组.DATE_FORMAT()格式化日期.ORDER BY排序.COUNT()统计行 ...

  10. Delphi Android拍照报错

    打开拍照提示以上错误,解决方式