DP在字符匹配上的实现
在此保存下近段时间做的DP在字符匹配上的实现的题目
对于不同的字符串来说,2者只能不断将下标往后推移来实现匹配从而得到的最大匹配数
如 abcd 和 dcba 这个最大匹配数只能为1,因为两个d匹配后,在第一个字符串中是不能再拿前面的字符进行匹配的(当然你要是匹配a,b,c也是一样的道理)
对于每一道题目若想不断找到那个匹配成功的字符的话,我们需要一个函数不断递归找到前一个匹配成功的字符,这里引进一个T[N][N]的标志位来帮助我们判断何时进行递归
这里的题都是有关这个的形式
1.POJ 1458http://vjudge.net/problem/viewProblem.action?id=17083
这就是最基本的2个字符串的最大匹配数
dp[i][j]代表前一个字符串取i位,后一个字符串取j位时得到的最大匹配数
DP 方程:dp[i][j]={dp[i-1][j-1]+1,a[i]==b[j] | max(dp[i-1][j],dp[i][j-1])}
自己一开始在做题时写成了dp[i][j]={dp[i-1][j-1]+1,a[i]==b[j] | max(dp[i][j],dp[i-1][j-1])}
这样仔细想想很容易发现dp[i][j]=max(dp[i][j],dp[i-1][j-1]) 只是考虑了一部分的状态,而且本身在i,j2次循环时dp[i][j]也只出现了一次,也就是说只
进行了一次赋值操作,根本就不会进行更新操作
dp[i][j]=max(dp[i-1][j],dp[i][j-1])却可以在循环中不断将前面得到的最大值赋给后面;
for(int j=1;j<=lb;j++)
if(a[i-1]==b[j-1])
{
dp[i][j]=max(dp[i][j],dp[i-1][j-1]+1);
}
else dp[i][j]=max(dp[i][j-1],dp[i-1][j]);//这一段要注意,这是为了匹配到2段数组均在小于它的情况下所得
}
总代码如下:
#include <cstdio>
#include <cstring>
using namespace std;
#define N 1001
#define max(a,b) a>b?a:b;
int dp[N][N];
char a[N],b[N]; int main()
{
while(scanf("%s%s",a,b)!=EOF){
memset(dp,,sizeof(dp));
int la=strlen(a),lb=strlen(b);
for(int i=;i<=la;i++){
for(int j=;j<=lb;j++)
if(a[i-]==b[j-])
{
dp[i][j]=max(dp[i][j],dp[i-][j-]+);
}
else dp[i][j]=max(dp[i][j-],dp[i-][j]);//这一段要注意,这是为了匹配到2段数组均在小于它的情况下所得
}
printf("%d\n",dp[la][lb]);
}
return ;
}
但是我们想要不断找到那个匹配成功的字符的话,我们需要一个函数不断递归找到前一个匹配成功的字符,这里引进一个T[N][N]的标志位来帮助我们判断何时进行递归
这是递归函数
void output(int len1,int len2)
{
if(len1==||len2==) return;
if(T[len1][len2]==){
output(len1-,len2-);
printf("%c\n",a[len1-]);
}
else if(T[len1][len2]==) output(len1-,len2);
else output(len1,len2-);
}
这是T[][]在进行寻找最长子序列时进行赋值
在第二题中要用到这个方法,就不再详述
2.POJ 2250
http://vjudge.net/problem/viewProblem.action?id=17325
这与上面不同的是这是给2堆字符串,找其中尽可能多的相同的字符串并输出
这里需要用到output(int,int)进行递归调用
总代码如下:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <string>
using namespace std;
#define N 103
string a[N],b[N];
int dp[N][N],T[N][N],len1,len2,cnt=; void match(int len1,int len2)
{
memset(dp,,sizeof(dp));
for(int i=;i<=len1;i++)
{
for(int j=;j<=len2;j++)
{
if(a[i-]==b[j-]) dp[i][j]=dp[i-][j-]+,T[i][j]=;
else{
if(dp[i-][j]>dp[i][j-]) dp[i][j]=dp[i-][j],T[i][j]=;
else dp[i][j]=dp[i][j-],T[i][j]=;
}
}
}
}
void output(int m,int n)
{
if(m==||n==) return;
if(T[m][n]==){
output(m-,n-);
cnt++;
if(cnt==dp[len1][len2])
cout<<a[m-]<<endl;
else cout<<a[m-]<<' ';
}
else if(T[m][n]==) output(m-,n);
else output(m,n-);
}
int main()
{
string str;
while(cin>>str)
{
len1=,len2=;
a[]=str;
while(cin>>str&&str.at()!='#'){
a[len1++]=str;
}
while(cin>>str&&str.at()!='#'){
b[len2++]=str;
}
match(len1,len2);
output(len1,len2);
//cout<<dp[len1][len2]<<endl;
}
return ;
}
DP在字符匹配上的实现的更多相关文章
- 《LeetBook》leetcode题解(10): Regular Expression Matching——DP解决正则匹配
我现在在做一个叫<leetbook>的免费开源书项目,力求提供最易懂的中文思路,目前把解题思路都同步更新到gitbook上了,需要的同学可以去看看 书的地址:https://hk029.g ...
- 【编程题目】有 n 个长为 m+1 的字符串,如果某个字符串的最后 m 个字符与某个字符串的前 m 个字符匹配...
37.(字符串)有 n 个长为 m+1 的字符串,如果某个字符串的最后 m 个字符与某个字符串的前 m 个字符匹配,则两个字符串可以联接,问这 n 个字符串最多可以连成一个多长的字符串,如果出现循环, ...
- (转)MySQL数据表中带LIKE的字符匹配查询
MySQL数据表中带LIKE的字符匹配查询 2014年07月15日09:56 百科369 MySQL数据表中带LIKE的字符匹配查询 LIKE关键字可以匹配字符串是否相等. 如果字段的值与指定的 ...
- leetcode 44 字符匹配
题意:s是空串或包含a-z字母: p为包含a-z字母或?或 * (其中*可以匹配任意字符串包括空串,?可以匹配任意字符). 思路: 1)特殊情况:当s为空串时,p为连续 * 时,则连续 * 的位置都为 ...
- 请实现一个函数用来匹配包括'.'和'*'的正则表达式。模式中的字符'.'表示任意一个字符,而'*'表示它前面的字符可以出现任意次(包含0次)。 在本题中,匹配是指字符串的所有字符匹配整个模式。例如,字符串"aaa"与模式"a.a"和"ab*ac*a"匹配,但是与"aa.a"和"ab*a"均不匹配
// test20.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> #include< ...
- KMP算法——字符匹配
暴力匹配: 假设现在我们面临这样一个问题:有一个文本串S,和一个模式串P,现在要查找P在S中的位置,怎么查找呢? 如果用暴力匹配的思路,并假设现在文本串S匹配到 i 位置,模式串P匹配到 j 位置, ...
- Python 基础之正则之一 单字符,多字符匹配及开头结尾匹配
一.正则表达式之单个字符匹配 格式:lst = re.findall(正则表达式,要匹配的字符串)预定义字符集 匹配内容 .匹配任意字符,除了换行符\n \d匹配数字 \D匹配非数字 \w匹配字母或数 ...
- Java 字符终端上获取输入三种方式
http://blog.csdn.net/hongweigg/article/details/14448731 在Java 字符终端上获取输入有三种方式: 1.java.lang.System.in ...
- JavaScript正则表达式模式匹配(1)——基本字符匹配
var pattern=/g..gle/; //点符号表示匹配除了换行符外的任意字符 var str='g78gle'; alert(pattern.test(str)); var pattern=/ ...
随机推荐
- vs2013转为vs2010项目
1.首先用记事本之类的工具打开.sln文件 打开后会看到如下信息 Format Version 12.00 就是指VS2013 VisualStudioVersion = 12.0.21005.1 指 ...
- AJPFX总结string类和简单问题
String表示字符串,所谓字符串,就是一连串的字符;String是不可变类,一旦String对象被创建,包含在对象中的字符序列(内容)是不可变的,直到对象被销毁://一个String对象的内容不能变 ...
- BZOJ1132: [POI2008]Tro(叉积 排序)
题意 世上最良心题目描述qwq 平面上有N个点. 求出所有以这N个点为顶点的三角形的面积和 N<=3000 Sol 直接模拟是$n^3$的. 考虑先枚举一个$i$,那么我们要算的就是$\sum_ ...
- 关于Java虚拟机
先占个坑,可以参考以下两篇文档来进行初步的学习 http://www.cnblogs.com/fingerboy/p/5456371.html http://www.importnew.com/244 ...
- grep的几个参数
-a 在二进制问就爱你中,以文本方式进行搜索 -c 计算找到搜索字符串的次数 -i 忽略大小写 -n 输出行号 -v 反向选择,即没有显示搜索字符串内容的那一行 grep -n '\.$' file ...
- codevs 2600 13号星期几?
时间限制: 1 s 空间限制: 8000 KB 题目等级 : 黄金 Gold 题目描述 Description 从1900年1月1日(星期一) 开始经过的n年当中,每个月的13号这一天是星期一.星 ...
- 推荐一个以动画效果显示github提交记录的黑科技工具:Gource
程序员每天都会使用到git的一系列命令.其中用git log命令可以查看提交历史记录: 今天Jerry给大家推荐一款视觉效果非常酷炫的工具,名叫Gource,是一个能够将git代码仓库的提交历史以动画 ...
- SAP成都研究院姚瑶:软件质量保证工作的变迁
大家好,我是来自SAP成都研究院Revenue Cloud 团队的质量工程师 , yoyo.很高兴可以和大家分享我个人的工作体会.每个团队都有QE(Quality Engineer), 相信大家对QE ...
- sqlserver 数据库主外键关联错误
话题引入: 在建立主外键关系时,系统提示表"table2"中的列与现有的主键或UNIQUE约束不匹配 原因: 数据库表中只有一个主键,这个主键可以是多个列共同组成.所以table2 ...
- 进程池_Pool
当需要创建子进程数量不多的时候,可以直接利用multiprocessing中的Process动态生成多个进程 但是如果是成百甚至上千个任务,手动地创建它的工作量很大,此时就可以利用到multiproc ...