热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Codeforces494BObsessiveString题解(KMP+DP)

题目:CF494B.题目大意:给定两个字符串SSS和TTT,在SSS中抽取几个不相交的子串,使得TTT均是这些子串的子串&#

题目:CF 494 B.
题目大意:给定两个字符串SSSTTT,在SSS中抽取几个不相交的子串,使得TTT均是这些子串的子串,问有多少种方案.
1≤∣S∣,∣T∣≤1051\leq |S|,|T|\leq 10^51S,T105.

仔细一看感觉题目好像比较难,但看完题解后感觉这也是一道比较容易的套路题,KMP与DP结合的题做的少啊…

首先很容易想到第一个串的子串必须包含第二个串,就很自然想到用KMP将第二个串在第一个串中出现的位置都找出来,那么所有的子串都必须包含这些位置.

考虑计算出一个数组val[i]val[i]val[i]表示第iii个字符前最近的匹配点(第iii个点不一定要匹配),发现这个数组可以很容易用KMP求出匹配点后稍微处理一下求出.

考虑应用valvalval数组,设f[i]f[i]f[i]表示到前iii个字符的方案数(所选子串中不一定要包含第iii个字符),然后很容易发现这个状态的转移有两种情况:
1.去掉第iii个字符,也就是f[i−1]f[i-1]f[i1].
2.找到最近的匹配点val[i]val[i]val[i],考虑f[1..val[i]−1]f[1..val[i]-1]f[1..val[i]1],发现可以只选串[val[i]..i][val[i]..i][val[i]..i]或者在f[1..val[i]−1]f[1..val[i]-1]f[1..val[i]1]后面直接加入这个串.

所以转移就是:
f[i]=f[i−1]+val[i]+∑j=1val[i]−1f[j]f[i]=f[i-1]+val[i]+\sum_{j=1}^{val[i]-1}f[j] f[i]=f[i1]+val[i]+j=1val[i]1f[j]

转移出来之后,发现答案就是f[n]f[n]f[n]了,所以直接输出就好.

代码如下:

#includeusing namespace std;#define Abigail inline void
typedef long long LL;const int N=100000;
const LL mod=1000000007;char s[N+9],t[N+9];
int n,m,nxt[N+9];
LL f[N&#43;9],sum[N&#43;9],val[N&#43;9];void add(LL &a,LL b){a&#43;&#61;b;while (a>&#61;mod) a-&#61;mod;}void self_mate(){int j&#61;0;nxt[1]&#61;0;for (int i&#61;2;i<&#61;m;&#43;&#43;i){while (t[i]^t[j&#43;1]&&j>0) j&#61;nxt[j];if (t[i]&#61;&#61;t[j&#43;1]) &#43;&#43;j;nxt[i]&#61;j;}
}void mate(){int j&#61;0;for (int i&#61;1;i<&#61;n;&#43;&#43;i){while (s[i]^t[j&#43;1]&&j>0) j&#61;nxt[j];if (s[i]&#61;&#61;t[j&#43;1]) &#43;&#43;j;if (j&#61;&#61;m){val[i]&#61;i-m&#43;1;j&#61;nxt[j];}}
}void solve(){for (int i&#61;1;i<&#61;n;&#43;&#43;i)if (val[i]&#61;&#61;0) val[i]&#61;val[i-1];for (int i&#61;1;i<&#61;n;&#43;&#43;i){f[i]&#61;f[i-1];if (val[i]) add(f[i],sum[val[i]-1]&#43;val[i]);add(sum[i],sum[i-1]&#43;f[i]);}
}Abigail into(){scanf("%s",s&#43;1);scanf("%s",t&#43;1);n&#61;strlen(s&#43;1);m&#61;strlen(t&#43;1);
}Abigail work(){self_mate();mate();solve();
}Abigail outo(){printf("%I64d\n",f[n]);
}int main(){into();work();outo();return 0;
}


推荐阅读
  • 在1995年,Simon Plouffe 发现了一种特殊的求和方法来表示某些常数。两年后,Bailey 和 Borwein 在他们的论文中发表了这一发现,这种方法被命名为 Bailey-Borwein-Plouffe (BBP) 公式。该问题要求计算圆周率 π 的第 n 个十六进制数字。 ... [详细]
  • 本文介绍如何手动实现一个字符串连接函数,该函数不依赖于C语言的标准字符串处理函数,如strcpy或strcat。函数原型为void concatenate(char *dest, char *src),其主要作用是将源字符串src追加到目标字符串dest的末尾。 ... [详细]
  • c语言二元插值,二维线性插值c语言
    c语言二元插值,二维线性插值c语言 ... [详细]
  • 本文详细介绍了如何在循环双链表的指定位置插入新元素的方法,包括必要的步骤和代码示例。 ... [详细]
  • 二维码的实现与应用
    本文介绍了二维码的基本概念、分类及其优缺点,并详细描述了如何使用Java编程语言结合第三方库(如ZXing和qrcode.jar)来实现二维码的生成与解析。 ... [详细]
  • 本文通过C++语言实现了一个递归算法,用于解析并计算数学表达式的值。该算法能够处理加法、减法、乘法和除法操作。 ... [详细]
  • 字符串中特定模式出现次数的计算方法
    本文详细探讨了如何高效地计算字符串中特定模式(如'pat')的出现次数,通过实例分析与算法解析,帮助读者掌握解决此类问题的方法。 ... [详细]
  • Go从入门到精通系列视频之go编程语言密码学哈希算法(二) ... [详细]
  • 本题要求计算一组正整数的最小公倍数(LCM)。输入包括多组测试数据,每组数据首先给出一个正整数n,随后是n个正整数。 ... [详细]
  • 使用QT构建基础串口辅助工具
    本文详细介绍了如何利用QT框架创建一个简易的串口助手应用程序,包括项目的建立、界面设计与编程实现、运行测试以及最终的应用程序打包。 ... [详细]
  • 线段树详解与实现
    本文详细介绍了线段树的基本概念及其在编程竞赛中的应用,并提供了一个具体的线段树实现代码示例。 ... [详细]
  • 本文详细介绍了C++中的构造函数,包括其定义、特点以及如何通过构造函数进行对象的初始化。此外,还探讨了转换构造函数的概念及其在不同情境下的应用,以及如何避免不必要的隐式类型转换。 ... [详细]
  • 从键盘输入年、月、日,要求输出当前日期为当年的第多少天。今天凯凯君又去参加了笔试,碰到了这样一个题目,从键盘输入年、月、日,要求输出当前日期为当年的第多少天。面对这个题目你首先想到 ... [详细]
  • 本文介绍了一种在ZC公司的员工评估系统中,如何根据动态设置的评分指标,在后台查询时动态生成并显示数据表的方法。该方法确保了评分指标与被评人员信息的有效整合。 ... [详细]
  • 1#include2#defineM1000103#defineRGregister4#defineinf0x3f3f3f3f5usingnamespacestd;6boolrev ... [详细]
author-avatar
liuleyi
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有