Codeforces 452E Three Strings(后缀自动机)
上学期很认真地学了一些字符串的常用工具,各种 suffix structre,但是其实对后缀自动机这个部分是理解地不太透彻的,以致于看了师兄A这题的代码后,我完全看不懂,于是乎重新看回一些学习后缀自动机的博客和代码,重新理解了一些概念,感觉自己对于后缀自动机的理解深了一些。
先说说题意,给了三个串,问对于长度为x的串,满足(i1,i2,i3)(s1[i1..i1+k-1]==s2[i2...i2+k-1]==s3[i3+i3+k-1])有多少组,要输出全部的组数。
师兄的代码的做法是这样的,插入三个串到后缀自动机上,中间用不同的分隔符隔开。然后对于每个状态求它的right集合的大小,求的是该字符串在三个串里的right集合的大小。具体的处理方法好像是这样的,因为有了分隔符,所以如果存在到达分隔符的后继,说明该状态是当前字符串的1个后缀,初始化为1,然后就dfs一遍,将right集合加回去。 此时对于每个状态right[0]*right[1]*right[2]即是该集合表示的字符串的三元对的大小,那么这样的三元对可以更新多少呢? 由后缀自动机的知识可以知道,对于状态s,它可以更新长度为min(s),max(s)的大小,max(s)就是下面的val,而后缀自动机的性质是min(s)=max(pre)-1。所以对于每个三元对段更,由于是先更新再询问,所以可以先打上标记,然后再回加,这样就可以求出正确的答案了。下面贴一记从师兄那里学习到的代码
#pragma warning(disable:4996)
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cmath>
#include <vector>
#include <string>
using namespace std; #define mod 1000000007
#define ll long long
#define maxn 610000 struct State{
State *suf, *go[30];
int val;;ll right[3];
State() :suf(0), val(0){
memset(go, 0, sizeof(go));
}
}*root,*last; State statePool[maxn], *cur; void init(){
cur = statePool;
root = last = cur++;
} void extend(int w){
State *p = last, *np = cur++;
np->val = p->val + 1;
while (p&&!p->go[w]) p->go[w] = np, p = p->suf;
if (!p) np->suf = root;
else{
State *q = p->go[w];
if (p->val + 1 == q->val) np->suf = q;
else{
State *nq = cur++;
memcpy(nq->go, q->go, sizeof q->go);
nq->val = p->val + 1;
nq->suf = q->suf;
q->suf = nq;
np->suf = nq;
while (p&&p->go[w] == q){
p->go[w] = nq, p = p->suf;
}
}
}
last = np;
} char str[maxn]; int n;
bool vis[maxn]; void dfs(State *u){
if (vis[u - statePool]) return;
if (!vis[u - statePool]){
vis[u - statePool] = true;
for (int i = 0; i < 3; i++){
if (u->go[26 + i]) u->right[i] = 1;
}
for (int i = 0; i < 26; i++){
if (u->go[i]) {
dfs(u->go[i]);
for (int k = 0; k < 3; k++){
u->right[k] = (u->right[k] + u->go[i]->right[k]) % mod;
}
}
}
}
} ll num[maxn]; int main()
{
init(); int n = 1e8;
scanf("%s", str);
n = min(n, int(strlen(str)));
for (int i = 0; str[i]; i++) extend(str[i] - 'a');
extend(26); scanf("%s", str);
n = min(n, int(strlen(str)));
for (int i = 0; str[i]; i++) extend(str[i] - 'a');
extend(27); scanf("%s", str);
n = min(n, int(strlen(str)));
for (int i = 0; str[i]; i++) extend(str[i] - 'a');
extend(28);
dfs(root); int tot = cur - statePool; for (int i = 1; i < tot; i++){
State *u = &statePool[i];
ll tmp = u->right[0] % mod * u->right[1] % mod*u->right[2] % mod;
num[u->val] = (num[u->val] + tmp) % mod;
num[u->suf->val] = ((num[u->suf->val] - tmp) % mod + mod) % mod;
} for (int i = tot-2; i >= 1; i--){
num[i] = (num[i] + num[i + 1]) % mod;
}
for (int i = 1; i <= n; i++){
if (i>1) printf(" ");
printf("%I64d", num[i]);
}
puts("");
}
Codeforces 452E Three Strings(后缀自动机)的更多相关文章
- Codeforces 235C Cyclical Quest - 后缀自动机
Some days ago, WJMZBMR learned how to answer the query "how many times does a string x occur in ...
- HDU 6208 The Dominator of Strings 后缀自动机
The Dominator of Strings Time Limit: 3000/3000 MS (Java/Others) Memory Limit: 65535/32768 K (Java ...
- Codechef2015 May - Chef and Strings (后缀自动机)
用后缀自动机统计出出现1~n次的串的数量f[i] 对于ans[k]=sigma(f[i]*C(i,k)) i>=k ; mo=; ..maxn] of dword; nt:..maxn,'a'. ...
- Codeforces 452E Three strings 字符串 SAM
原文链接https://www.cnblogs.com/zhouzhendong/p/CF542E.html 题目传送门 - CF452E 题意 给定三个字符串 $s1,s2,s3$ ,对于所有 $L ...
- CF 149E Martian Strings 后缀自动机
这里给出来一个后缀自动机的题解. 考虑对 $s$ 的正串和反串分别建后缀自动机. 对于正串的每个节点维护 $endpos$ 的最小值. 对于反串的每个节点维护 $endpos$ 的最大值. 这两个东西 ...
- [Codeforces 452E] Three Strings
[题目链接] https://codeforces.com/contest/452/problem/E [算法] 构建后缀数组 用并查集合并答案即可 时间复杂度 : O(NlogN) [代码] #in ...
- CF 452E. Three strings(后缀数组+并查集)
传送门 解题思路 感觉这种题都是套路之类的??首先把三个串并成一个,中间插入一些奇怪的字符,然后跑遍\(SA\).考虑按照\(height\)分组计算,就是每个\(height\)只在最高位计算一次, ...
- Codeforces.700E.Cool Slogans(后缀自动机 线段树合并 DP)
题目链接 \(Description\) 给定一个字符串\(s[1]\).一个字符串序列\(s[\ ]\)满足\(s[i]\)至少在\(s[i-1]\)中出现过两次(\(i\geq 2\)).求最大的 ...
- 字符串(后缀自动机):Codeforces Round #129 (Div. 1) E.Little Elephant and Strings
E. Little Elephant and Strings time limit per test 3 seconds memory limit per test 256 megabytes inp ...
随机推荐
- NOJ1103-全排列
全排列 时间限制(普通/Java) : 1000 MS/ 3000 MS 运行内存限制 : 65536 KByte总提交 : 1148 测试通过 : 302 ...
- 算法系列4《Luhn》
Luhn算法由IBM的Hans Peter Luhn发明,又称为"模10"算法,是一种简单的校验和算法,用来验证识别号,一般会被用于身份证号码,信用卡号.IMEI号.社会保险号的验 ...
- JAVA基础-子类继承父类实例化对象过程
之前在项目中碰到这样一个问题: 类B继承了类A,B在实例化的时候,A的构造方法中调用了B的某个方法,并且B的方法中对B的成员属性进行了初始化,然后最后得到的B对象的成员属性为空. 代码场景如下: 这里 ...
- c++编程规范的纲要和记录
这是一本好书, 可以让你认清自己对C++的掌握程度.看完之后,给自己打分,我对C++了解多少? 答案是不足20分.对于我自己是理所当然的问题, 就不提了, 记一些有启发的条目和细节: (*号表示不能完 ...
- windows phone 豆瓣api的封装
利用周末的时候,重新封装一下豆瓣的api,就当是练手吧!其实现在网上好用的api很多,在这个demo里面基本上已经将整体框架搭建起来,本来想继续完善下去的.但是其实accesstoken的时候,一直拿 ...
- 关于DataSource的一些记录
今天看WWDC的232_hd_advanced_user_interfaces_with_collection_views,里面花了一般的时间来讲如何合理的设计程序的datesource,将的很有道理 ...
- ultraedit高亮显示设置
ultraedit高亮显示设置 | 浏览:2333 | 更新:2014-02-20 23:05 1 2 3 4 5 6 7 分步阅读 百度经验:jingyan.baidu.com 写代码的人对ultr ...
- 将inline、template声明和定义在头文件中
如果要在要在源文件(a.cpp)中内联的展开一个函数(fun),则该源文件(a.cpp)中必须包含此函数(fun)的定义.如果要在多个文件中内联的展开fun,则在所有的源文件中都必须包含fun的定义. ...
- cocos2dx中的场景和使用方法
1.一个游戏中有且只有一个导演,但是至少有一个场景 2.场景是游戏元素节点数的根节点,也可以理解为该场景下的渲染树的根节点 3.场景是一个容器,包含了该场景下的所有游戏元素,比如层,精灵 4.场景是导 ...
- pspo
一.项目计划总结: 周活动总结表 姓名: 日期:3.12.2015 日期 任务 听课 编写程序 阅读课本 准备考试 日总计 周日 周一 周二 周三 10:00- ...