[NOIP2015] 子串(dp)
题目描述
有两个仅包含小写英文字母的字符串 A 和 B。现在要从字符串 A 中取出 k 个互不重叠的非空子串,然后把这 k 个子串按照其在字符串 A 中出现的顺序依次连接起来得到一 个新的字符串,请问有多少种方案可以使得这个新串与字符串 B 相等?注意:子串取出 的位置不同也认为是不同的方案。
输入输出格式
输入格式:
输入文件名为 substring.in。
第一行是三个正整数 n,m,k,分别表示字符串 A 的长度,字符串 B 的长度,以及问
题描述中所提到的 k,每两个整数之间用一个空格隔开。 第二行包含一个长度为 n 的字符串,表示字符串 A。 第三行包含一个长度为 m 的字符串,表示字符串 B。
输出格式:
输出文件名为 substring.out。 输出共一行,包含一个整数,表示所求方案数。由于答案可能很大,所以这里要求[b]输出答案对 1,000,000,007 取模的结果。[/b]
输入输出样例
6 3 1
aabaab
aab
2
6 3 2
aabaab
aab
7
6 3 3
aabaab
aab
7
说明

对于第 1 组数据:1≤n≤500,1≤m≤50,k=1;
对于第 2 组至第 3 组数据:1≤n≤500,1≤m≤50,k=2;
对于第 4 组至第 5 组数据:1≤n≤500,1≤m≤50,k=m;
对于第 1 组至第 7 组数据:1≤n≤500,1≤m≤50,1≤k≤m;
对于第 1 组至第 9 组数据:1≤n≤1000,1≤m≤100,1≤k≤m;
对于所有 10 组数据:1≤n≤1000,1≤m≤200,1≤k≤m。
- 比较厉害的dp题,搞了好长时间才搞出来的。
- 而且还有各种坑,卡时间,卡空间。。。。。。
- 不会讲很多,只是简单的说一下思路。
- f[k][i][j]表示分了k段,第一个串取了前i个,第二个串已经构成了前j个的方案数。显然,当s[i]!=ss[j]时f[k][i][j]=0;
- f[k][i][j]=∑f[k-1][L][j-1] 其中s[i]==ss[j] && s[i-1]!=ss[j-1] && 0<L<i
- f[k][i][j]=f[k-1][L][j-1]+f[k][i-1][j-1],其中s[i]==ss[j] && s[i-1]==ss[j-1] && 0<L<i
- 以上是dp的思路,不过这样做只能得到70分,其余三个点会tle。
- 思考一下时间浪费在哪里?
- 通过dp方程可知时间复杂度为O(n2mk),每一次需要通过枚举来找到∑的值。
- 优化一下就是利用前缀和的思想,每次用t数组记录∑的值,这样可以优化到O(nmk),,时间复杂度上比较理想。
- 但是还是无法A掉该题,只能得90分,因为空间炸了。如果开1000*200*200*2的数组,大概需要300MB的空间。。。。
- 不难发现方程里只有i,i-1在被调用,以前的空间都已经失效了,所以就用滚动数组吧。
- 空间和时间都可以符合本题的要求,可以得到100分。
#include <cstdio>
#include <cstring>
#include <algorithm>
#define mod 1000000007
using namespace std; int n,m,K;
char s[],ss[];
long long f[][][];
long long t[][][];
long long ans; int main() {
scanf("%d%d%d",&n,&m,&K);
scanf("%s%s",s+,ss+);
f[][][]=t[][][]=;
for (int i=; i<=n; i++) t[][i][]=;
for (int k=; k<=K; k++) {
memset(f[k&],,sizeof f[k&]);
memset(t[k&],,sizeof t[k&]);
for (int i=; i<=n; i++) {
for (int j=; j<=m; j++) {
if (s[i]==ss[j]) {
f[k&][i][j]=(long long)t[(k+)&][i-][j-];
if (s[i-]==ss[j-]) f[k&][i][j]=(long long)(f[k&][i][j]+f[k&][i-][j-]+mod)%mod;
}
t[k&][i][j]=(long long)((t[k&][i][j]+f[k&][i][j]+mod)%mod+t[k&][i-][j]+mod)%mod;
}
}
}
for (int i=; i<=n; i++) ans=(long long)((ans+f[K&][i][m]+mod)%mod);
printf("%lld",ans);
return ;
}
[NOIP2015] 子串(dp)的更多相关文章
- $[NOIp2015]$ 子串 $dp$
\(Sol\) 不知道为啥看起来就很\(dp\)的亚子.我们关心的只有\(A\)串当前用到哪一个,\(B\)串已经匹配到哪个位置,已经匹配的被分成了多少段.所以设\(f_{i,j,k,0/1}\)表示 ...
- luogu2679 [NOIp2015]子串 (dp)
设f[i][j][k][b]表示在A串第i位.这是第j组.B串第k位.i号选不选(b=0/1) 那么就有$f[i][j][k][1]=(A[i]==B[k])*(f[i-1][j-1][k][0]+f ...
- NOIP2015子串[序列DP]
题目背景 无 题目描述 有两个仅包含小写英文字母的字符串 A 和 B.现在要从字符串 A 中取出 k 个互不重 叠的非空子串,然后把这 k 个子串按照其在字符串 A 中出现的顺序依次连接起来得到一 个 ...
- LOJ2424 NOIP2015 子串 【DP】*
LOJ2424 NOIP2015 子串 LINK 题目大意是给你两个序列,在a序列中选出k段不重叠的子串组成b序列,问方案数 首先我们不考虑相邻的两段,把所有相邻段当成一段进行计算 然后设dpi,j, ...
- P2679 子串 DP
P2679 子串 DP 从字符串A中取出\(k\)段子串,按原顺序拼接,问存在多少个方案使拼接的字符串与字符串B相同 淦,又是这种字符串dp 设状态\(ans[i][j][k]\)表示A串位置\(i\ ...
- NOIP2015 子串 (DP+优化)
子串 (substring.cpp/c/pas) [问题描述] 有两个仅包含小写英文字母的字符串 A 和 B.现在要从字符串 A 中取出 k 个 互不重 叠 的非空子串,然后把这 k 个子串按照其在字 ...
- [DP][NOIP2015]子串
子串 题目描述 有两个仅包含小写英文字母的字符串 A 和 B. 现在要从字符串 A 中取出 k 个 互不重叠 的非空子串, 然后把这 k 个子串按照其在字符串 A 中出现的顺序依次连接起来得到一个新的 ...
- NOIP2015 子串
#149. [NOIP2015]子串 有两个仅包含小写英文字母的字符串 AA 和 BB. 现在要从字符串 AA 中取出 kk 个互不重叠的非空子串,然后把这 kk 个子串按照其在字符串 AA 中出现的 ...
- 【uoj149】 NOIP2015—子串
http://uoj.ac/problem/149 (题目链接) 题意 给出两个字符串A.B,问从A中取出k个互不重叠的子串按顺序组成B的方案数. Solution 一看这种题目就是字符串dp,字符串 ...
随机推荐
- 查询oracle版本信息
select * from v$version;select * from v$instance;select * from nls_database_parameters;select * from ...
- [DNS][转]EDNS
随着业务的复杂化和多样化,RFC1035中定义的DNS消息格式和它支持的消息内容已经不足以满足一些DNS服务器的需求,于是,RFC2671 中提出了一种扩展DNS机制EDNS(Extension Me ...
- 【1】springmvc4 + servlet3 零配置全注解入门项目helloword
自从servlet3.0支持注解,使得javaweb项目使用全注解成为可能. 注解化将成为javaweb的发展方向.包括spring去年推出的spring-boot也是全部使用注解. 代码:https ...
- pb自动注册ole控件
方法一: 1.手工注册OCX控件 将该控件随程序一起发布,然后,将此文件拷到windows\system,或者直接放在本运行目录,然后执行dos命令,run( "regsvr32 *. ...
- ios开发之网络php
接着前面的学习,几天上午学习了数据库网络值php用户的注册与登录,感觉代码与ios上的oc太相似了,因此学习下来没什么障碍了,下面是代码: 首先是javascript.html文件中: <!DO ...
- Windows7+VS2010下OpenGL的环境配置
http://johnhany.net/2014/01/environment-for-opengl-with-vs2010/ OpenGL(Open Graphics Library)是一个开放的. ...
- 模板volist自增变量
- UE4 代码编写细节:静态变量
Note:因为在切换关切时,会GC掉所有GameThread线程下的Object类,如果Static是UOBject 请调用AddToRoot函数 当然如果你的UObject子类Object是在自己 ...
- 电量/信号的显示 imageLevel
通过imageLevel可以对1个imageView在不同的level值显示不同的图片,通常用于显示电量,wifi信号,蓝牙信号等信息: 1.编写xml ,在drawable目录 <level- ...
- Android上传图片到服务器
一.android需要导入异步请求的jar包 AsyncHttpClient public static void reg(final Context cont,Bitmap photodata,S ...