3508. 最长公共子串

​​​​​3508. 最长公共子串 - AcWing题库

给定两个字符串,求这两个字符串的不包含数字的最长公共子串的长度。

输入格式

共两行,每行一个由小写字母和数字构成的字符串。

输出格式

一个整数,表示给定两个字符串的不包含数字的最长公共子串的长度。

如果不存在满足要求的非空公共子串,则输出 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;
}

  • 13
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值