给定两个字符串,求这两个字符串的不包含数字的最长公共子串的长度。
输入格式
共两行,每行一个由小写字母和数字构成的字符串。
输出格式
一个整数,表示给定两个字符串的不包含数字的最长公共子串的长度。
如果不存在满足要求的非空公共子串,则输出 00。
数据范围
输入字符串的长度均不超过 1000010000。
输入样例:
ab123abccff abcfacb123
输出样例:
3
只要把两个字符串中的数字分别转换成两个互不相同的任意字符,就变为最长公共子串问题,不同于经典的最长公共子序列问题,子串不是子序列,子串要求连续,子序列则是可以按顺序挑选,但他们的思考方式都差不多,都可以通过二维数组 f[i][j] 表示 1~ai,1~bj且ai,bj为结尾的所有公共子串
那么状态转换方程就是 ,如果ai==bj且ai,bj都是字母,则f[i][j]=f[i-1][j-1]+1,否则 f[i][j]=0,
#include <bits/stdc++.h>
using namespace std;
int f[10001][10001];
int max(int a,int b) {
if(a>b) return a;
return b;
}
int main() {
string a,b;
cin>>a;
cin>>b;
int n=a.size();
int m=b.size();
for (int i = 0; i < n; i++) {
if (a[i] >= '0' && a[i] <= '9') {
a[i] = '#';
}
}
for (int i = 0; i < m; i++) {
if (b[i] >= '0' && b[i] <= '9') {
b[i] = '$';
}
}
int ans = -1;
for(int i=1; i<n; i++) {
for(int j=1; j<m; j++) {
if(a[i-1]==b[j-1])
f[i][j]=f[i-1][j-1]+1;
else f[i][j]= 0 ;
ans = max(ans,f[i][j]);
}
}
printf("%d",ans);
return 0;
}
此代码可以像01背包一样优化为一维,代码如下
#include <bits/stdc++.h>
using namespace std;
int f[10001];
int max(int a,int b) {
if(a>b) return a;
return b;
}
int main() {
string a,b;
cin>>a;
cin>>b;
int n=a.size();
int m=b.size();
for (int i = 0; i < n; i++) {
if (a[i] >= '0' && a[i] <= '9') {
a[i] = '#';
}
}
for (int i = 0; i < m; i++) {
if (b[i] >= '0' && b[i] <= '9') {
b[i] = '$';
}
}
int ans = -1;
for(int i=0; i<n; i++) {
for(int j=m-1; j>=0; j--) {
if(a[i]==b[j])
f[j]=f[j-1]+1;
else f[j]= 0 ;
ans = max(ans,f[j]);
}
}
printf("%d",ans);
return 0;
}
此题还能使用字符串哈希+map+二分实现,
ac代码:
#include<bits/stdc++.h>
using namespace std;
char a[10010], b[10010];
unsigned long long p[10010];
long long av[10010], bv[10010];
int k = 131;
long long getsvalue(int l, int r, long long v[]) {
return v[r] - v[l - 1] * p[r - l + 1];
}
int check(int x) {
map<unsigned long long, int>res;
for (int i = 1; i + x - 1 <= strlen(a+1); i++) {
int j = i + x - 1;
res[getsvalue(i, j, av)] = 1;
}
for (int i = 1; i + x - 1 <= strlen(b+1); i++) {
int j = i + x - 1;
if (res[getsvalue(i, j, bv)])
return 1;
}
return 0;
}
int main() {
p[0] = 1;
scanf("%s%s", a + 1, b + 1);
for (int i = 1; i <= strlen(a+1); i++) {
if (a[i] >= '0' && a[i] <= '9') {
a[i] = '#';
}
av[i] = av[i - 1] * k + a[i] ;
// av[i]=av[i]%mod;
}
for (int i = 1; i <= strlen(b+1); i++) {
if (b[i] >= '0' && b[i] <= '9') {
b[i] = '$';
}
bv[i] = bv[i - 1] * k + b[i] ;
// bv[i]=bv[i]%mod;
}
for (int i = 1; i <= max(strlen(a+1), strlen(b+1)); i++) {
p[i] = p[i - 1] * k;
// p[i]=p[i]%mod;
}
int l = 0, r = min(strlen(a+1),strlen(b+1));
while (l < r) {
int mid = l + r + 1 >> 1;
if (check(mid)) {
l = mid;
} else r = mid - 1;
}
cout << l;
}