![[vijos1982][NOIP2015]子串 [vijos1982][NOIP2015]子串](https://image.shishitao.com:8440/aHR0cHM6Ly9ia3FzaW1nLmlrYWZhbi5jb20vdXBsb2FkL2NoYXRncHQtcy5wbmc%2FIQ%3D%3D.png?!?w=700&webp=1)
Description
有两个仅包含小写英文字母的字符串和
。现在要从字符串
中取出
个互不重叠的非空子串,然后把这
个子串按照其在字符串
中出现的顺序依次连接起来得到一个新的字符串,请问有多少种方案可以使得这个新串与字符串
相等?注意:子串取出的位置不同也认为是不同的方案。
Input
第一行是三个正整数,分别表示字符串
的长度,字符串
的长度,以及问题描述中所提到的
,每两个整数之间用一个空格隔开。
第二行包含一个长度为的字符串,表示字符串
。
第三行包含一个长度为的字符串,表示字符串
。
Output
输出共一行,包含一个整数,表示所求方案数。
由于答案可能很大,所以这里要求输 出答案对取模的结果。
Sample Input
6 3 2
aabaab
aab
Sample Output
7
HINT
Solution
表示
中的
个子串与
相等的方案数(0表示不取
,1表示取)
滚动数组优化空间复杂度.
#include<cmath>
#include<ctime>
#include<stack>
#include<queue>
#include<cstdio>
#include<vector>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define M 205
#define N 1005
#define K 1000000007
using namespace std;
int f[][M][M][],n,m,k;
char a[N],b[M];
inline void init(){
scanf("%d%d%d%s%s",&n,&m,&k,a+,b+);
f[][][][]=;
for(int i=,p,q;i<=n;++i){
p=i&;q=p^;
f[p][][][]=;
for(int j=;j<=m;++j)
for(int l=;l<=k;++l){
f[p][j][l][]=(f[q][j][l][]+f[q][j][l][])%K;
if(a[i]==b[j]) f[p][j][l][]=((f[q][j-][l][]+f[q][j-][l-][])%K+f[q][j-][l-][])%K;
else f[p][j][l][]=;
}
}
printf("%d\n",(f[n&][m][k][]+f[n&][m][k][])%K);
}
int main(){
freopen("str.in","r",stdin);
freopen("str.out","w",stdout);
init();
fclose(stdin);
fclose(stdout);
return ;
}