「Codeforces 235C」Cyclical Quest
Posted -wallace-
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了「Codeforces 235C」Cyclical Quest相关的知识,希望对你有一定的参考价值。
Description
给定一个字符串 (S),以及 (m) 个询问。
每个询问给出一个字符串 (T),求 (T) 的所有循环同构在 (S) 中出现的次数总和。
Hint
- (1le nle 10^5)
- (1le |S| le 10^6)
- (1le sum|T| le 10^6)
Solution
后缀自动机。
一个比较简单的思路是,先对 (S) 建 SAM,顺便处理出 ( ext{end-pos}) 集大小。
然后对于每一个 (T),将其拆成 (|T|) 个字符串分别放在 SAM 上跑,把所有结果求和即可。
但这样一次复杂度为 (O(|T|^2)),这样的算法显然不可行。
我们将 (T) 复制一遍接在后面,记为 (R)。
注意到 (R) 中包含了所有 (T) 的循环同构的字符串,所以可以用 (R) 直接在 SAM 上跑。
如果当前匹配出的长度大于 (|T|),那么 强制失配。因为显然怎么循环都不会有这么长的串,而这个现象是复制所导致的。
为避免应旋转产生相同字符串对答案的影响,可以用一个标记数组。
时间,空间复杂度:(O(|S| + sum|T|)),(O(|S|))。(字符集大小记为常数)
Code
/*
* Author : _Wallace_
* Source : https://www.cnblogs.com/-Wallace-/
* Problem : Codeforces 235C Cyclical Quest
*/
#include <iostream>
#include <map>
#include <string>
using namespace std;
const int N = 1e5 + 5;
const int L = 1e6 + 5;
namespace SAM {
const int T = L << 1;
struct Node {
map<char, int> ch;
int link, len, size;
} t[T];
int last;
int total;
void extend_char(char c) {
int p = last, np = last = ++total;
t[np].len = t[p].len + 1;
for (; p && !t[p].ch[c]; p = t[p].link)
t[p].ch[c] = np;
if (!p) {
t[np].link = 1;
} else {
int q = t[p].ch[c];
if (t[p].len + 1 == t[q].len) {
t[np].link = q;
} else {
int nq = ++total;
t[nq] = t[q], t[nq].len = t[p].len + 1, t[nq].size = 0;
t[np].link = t[q].link = nq;
for (; p && t[p].ch[c] == q; p = t[p].link)
t[p].ch[c] = nq;
}
}
t[np].size = 1;
}
int b[T], c[T];
void init_data(string& s) {
last = total = 1;
for (string::iterator it = s.begin(); it != s.end(); it++)
extend_char(*it);
for (register int i = 1; i <= total; i++) ++c[t[i].len];
for (register int i = 1; i <= total; i++) c[i] += c[i - 1];
for (register int i = 1; i <= total; i++) b[c[t[i].len]--] = i;
for (register int i = total; i; i--) {
int x = b[i], f = t[x].link;
t[f].size += t[x].size;
}
}
int vis[T];
long long count(string& str, int time);
};
long long SAM::count(string& str, int time) {
long long ret = 0ll;
int x = 1, len = 0;
str = str + str;
for (register int i = 0; i < str.size(); i++) {
char c = str[i];
while (x && !t[x].ch[c]) x = t[x].link, len = t[x].len;
if (!x) x = 1, len = 0;
else x = t[x].ch[c], len++;
while (x && t[t[x].link].len >= str.size() / 2)
x = t[x].link, len = t[x].len;
if (len >= str.size() / 2 && vis[x] != time)
ret += t[x].size, vis[x] = time;
}
return ret;
}
signed main() {
ios::sync_with_stdio(false);
string str;
cin >> str;
SAM::init_data(str);
int m;
cin >> m;
for (register int time = 1; time <= m; time++) {
cin >> str;
cout << SAM::count(str, time) << endl;
}
return 0;
}
以上是关于「Codeforces 235C」Cyclical Quest的主要内容,如果未能解决你的问题,请参考以下文章
「Codeforces 235C」Cyclical Quest
Codeforces 235C. Cyclical Quest
Project Euler:Problem 61 Cyclical figurate numbers