CodeForces 494B Obsessive String ——(字符串DP+KMP)
这题的题意就很晦涩。题意是:问有多少种方法,把字符串s划分成不重叠的子串(可以不使用完s的所有字符,但是这些子串必须不重叠),使得t串是所有这些新串的子串。譬如第一个样例,"ababa"和"aba",共有5种方法:{aba}(前3个),{aba}(后3个),{abab},{baba},{ababa}。
先设s的长度为lena,t的长度为lenb。
做法是:用dp[i]表示到i为止,有几种方案数,所以最终答案是dp[lena]。然后考虑转移。首先dp[i]至少等于dp[i-1]。然后考虑把包含了一个t串的a[1~i]的后缀给取出来,不妨设目前的a[1~i]为*****abc,,t是"abc",然后新增的方法数有:如果右边包含t串的是"abc",那么左边的选择有(dp[i-3]+1)种(1是左边为空串的情况),类似的,如果右边包含t串的是"*abc",那么左边是(dp[i-4]+1)。那么累和一下即是,(dp[0]+1)+(dp[1]+1)+(dp[2]+1)+...+(dp[i-lenb]+1)。显然的,dp[0] = 0。那么这个式子就可以化简为sum(dp[1~i-lenb])+i-lenb+1。sum部分可以利用前缀和优化,如果假设pre[i]是dp[1~i]的和的话,那么sum部分变为pre[i-lenb],其余部分是i-lenb+1。假设val[i]表示i这个位置开始往前,至少包含了一个t串时的位置,例如串"*****abc",val[8] = 6,串"*****abc*",val[9] = 6。理解了val数组的含义以后,我们就可以发现如果i这个位置是t串的最后一个位置的话,那么val[i] = i-lenb+1,否则呢,val[i] = val[i-1]即可。我们可以用kmp把能够处理的val直接处理出来,其余的val通过递推得到。有了val数组以后,这个dp的转移就可以完全得到了:dp[i] = dp[i-1] + pre[val[i]-1] + val[i]。然后dp的过程中维护一下前缀和即可,最后需要注意要对1e9+7取模。
另外需要注意的是,如果说当前的串最后一个字符并不是t串的最后一个字符,例如"*****abc*",这样的话通过上面这个转移我们可以知道,右边的串也必须是以最后一个字符结尾的(实际上任何情况下右边这个串都是需要以最后一个字符结尾的,否则新增的这个字符就没有意义了,也就会出现和之前的方案重复的情况了)。这一点想清楚了,这个转移就没有任何的问题了。
最终的代码如下:
#include <stdio.h>
#include <algorithm>
#include <string.h>
using namespace std;
const int N = 1e5 + ;
const int mod = 1e9 + ; char a[N],b[N];
int lena,lenb,nxt[N];
int val[N];
int sum[N];
int dp[N];
void get_nxt()
{
nxt[] = ;
int j = ;
for(int i=;i<=lenb;i++)
{
while(j && b[j+] != b[i]) j = nxt[j];
if(b[j+] == b[i]) j++;
nxt[i] = j;
} j = ;
for(int i=;i<=lena;i++)
{
while(j && b[j+] != a[i]) j = nxt[j];
if(b[j+] == a[i]) j++;
if(j == lenb)
{
val[i] = i - lenb + ;
j = nxt[j];
}
}
} int main()
{
scanf("%s%s",a+,b+);
lena = strlen(a+);
lenb = strlen(b+);
get_nxt();
for(int i=;i<=lena;i++) if(!val[i]) val[i] = val[i-];
for(int i=;i<=lena;i++)
{
dp[i] = dp[i-];
if(val[i]) dp[i] = (dp[i] + sum[val[i]-] + val[i]) % mod;
sum[i] = (sum[i-] + dp[i]) % mod;
}
printf("%d\n",dp[lena]);
return ;
}
CodeForces 494B Obsessive String ——(字符串DP+KMP)的更多相关文章
- codeforces 825F F. String Compression dp+kmp找字符串的最小循环节
/** 题目:F. String Compression 链接:http://codeforces.com/problemset/problem/825/F 题意:压缩字符串后求最小长度. 思路: d ...
- Codeforces 494B Obsessive String
http://www.codeforces.com/problemset/problem/494/B 题意:给出两个串S,T,求有几种将S分成若干个子串,满足T都是这若干个子串的子串. 思路:f[n] ...
- [Codeforces-div.1 494B]Obsessive String
[CF-div.1 B]Obsessive String 题目大意 两个字符串\(S,T\),求划分方案数使得一个集合中两两划分不相交且划分都包含字符串\(T\) 试题分析 kmp先求出那个位置匹配. ...
- HDU3689 Infinite monkey theorem 无限猴子(字符串DP+KMP)
题目描述: 大概的意思就是根据无限猴子定理,无限只猴子坐在打字机旁瞎敲,总有一个能敲出莎士比亚文集.现在给你一个打字机和一只猴子,打字机的每个按钮(共n个)上的字母及猴子按下这个按钮的概率已知,而且猴 ...
- codeforces#1120C. Compress String(dp+后缀自动机)
题目链接: https://codeforces.com/contest/1120/problem/C 题意: 从前往后压缩一段字符串 有两种操作: 1.对于单个字符,压缩它花费$a$ 2.对于末尾一 ...
- codeforces#1183H. Subsequences(字符串dp)
题目链接: http://codeforces.com/contest/1183/problem/H 题意: 给出一个长度为$n$的字符串,得到$k$个子串,子串$s$的花费是$n-|s|$ 计算最小 ...
- Codeforces 235C Cyclical Quest 字符串 SAM KMP
原文链接https://www.cnblogs.com/zhouzhendong/p/CF235C.html 题目传送门 - CF235C 题意 给定一个字符串 $s$ ,多组询问,每组询问的形式为 ...
- [LeetCode] Scramble String 字符串 dp
Given a string s1, we may represent it as a binary tree by partitioning it to two non-empty substrin ...
- DP × KMP
几道用到KMP的DP题: hdu 5763 hdu 3689 hdu 3336 codeforces 494B codevs 3945 关于KMP的nx数组: 如果在本文中看见 ...
随机推荐
- Swiper 轮播插件 之 动态加载无法滑动
1.原因:轮播图未完全动态加载完成,即初始化 2.方法一:ajax链式编程 $.ajax({ type: "get", url: serviceURL + "/listB ...
- 奇妙的算法【9】YC每个小孩的糖果数,找公约数,最少硬币数
1,每个小孩的糖果数量是多少 有p个小孩,c个糖果,刚开始第1个小孩发一个糖果,第2个小孩发两个糖果,第p个小孩发p个糖果,如果糖果没有发完,就接着[注意]第1个小孩发p+1个糖果.....第p个小孩 ...
- 设计模式(四)——代理模式(Proxy)
代理模式的参与者有:一个约束.一个代理者.一个被代理者.一个调用者 代理模式的实现很简单:还是那个房子,对于开门这个操作,我更换了一个远程解锁的门,那么我就可以通过这个远程连接的服务器远程解锁,这样我 ...
- Oracle 多租户环境学习路线图
Category Topic Documentation Concepts Overview of CDBs and PDBs "Overview of the Multitenant Ar ...
- 算法之暴力破解和kmp算法 判断A字符串是否包含B字符串
我们都知道java中有封装好的方法,用来比较A字符串是否包含B字符串 如下代码,contains,用法是 str1.contains(str2), 这个布尔型返回,存在返回true,不存在返回fals ...
- 用js刷剑指offer(二叉树的镜像)
题目描述 操作给定的二叉树,将其变换为源二叉树的镜像. 输入描述: 二叉树的镜像定义:源二叉树 8 / \ 6 10 / \ / \ 5 7 9 11 镜像二叉树 8 / \ 10 6 / \ / \ ...
- vim 操作命令大全
转子:https://www.cnblogs.com/yangjig/p/6014198.html 和 https://blog.csdn.net/u010956473/article/detail ...
- vue 后台获取文件流导出excel文件
let params = { compStartTm: Date.parse(this.searchForm.compStartTm) / 1000, compEndTm: Date.parse(th ...
- Nginx中ngx_http_upstream_module模块
用于将多个服务器器定义成服务器器组,⽽而由 proxy_pass , fastcgi_pass 等指令进⾏行行引⽤用upstream backend {server backend1.example. ...
- asp.net中gridview控件的一些基本使用方法
[ 转自苏飞博客]共两篇 (1)菜单目录: GridView无代码分页排序GridView选中,编辑,取消,删除GridView正反双向排序GridView和下拉菜单DropDownList结合Gri ...