CodeForces 547E Mike and Friends AC自动机 主席树
题意:
给出\(n\)个字符串\(s_i\)和\(q\)个询问:
- \(l,r,k\):\(\sum\limits_{i=l}^{r}count(i, k)\),其中\(count(i,j)\)表示\(s_j\)作为子串在\(s_i\)中出现的次数
分析:
先不考虑查询中\(l,r\)的限制,考虑该字符串\(s_k\)在一个字符串集合中出现的次数。
先将这个字符串集合插入到一棵Trie数中,并且每经过一个节点就将其对应的\(val\)值加\(1\)。
这样\(s_k\)对应节点的\(val\)值就是以\(s_k\)为前缀的字符串的个数。
然而这还不够,接着构造出一棵fail树,也就是AC自动机中的fail指针构成的树。
这棵fail树中,父节点是子节点的后缀。
因为fail树中\(s_k\)是其子节点的后缀,所以再加上子树节点的\(val\)值。
这样就把问题转化为了求子树节点的权值之和:
利用DFS序将子树转为区间,然后用线段树维护区间和
然后考虑上题中的\(l,r\),可以将线段树可持久化或者离线查询加树状数组维护,而且离线应该是很快的。
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#include <map>
#include <set>
#include <queue>
using namespace std;
typedef long long LL;
typedef pair<int, int> PII;
#define PB push_back
#define PII pair<int, int>
#define REP(i, a, b) for(int i = a; i < b; i++)
#define PER(i, a, b) for(int i = b - 1; i >= a; i--)
#define ALL(x) x.begin(), x.end()
const int maxn = 200000 + 10;
const int nlogn = maxn * 40;
//persistant segment tree
int lch[nlogn], rch[nlogn], sum[nlogn];
int tot, root[maxn], p[maxn];
void update(int& rt, int pre, int L, int R, int p, int v = 1) {
rt = ++tot;
if(L == R) { sum[rt] = sum[pre] + v; return; }
int M = (L + R) / 2;
if(p <= M) { rch[rt] = rch[pre]; update(lch[rt], lch[pre], L, M, p, v); }
else { lch[rt] = lch[pre]; update(rch[rt], rch[pre], M+1, R, p, v); }
sum[rt] = sum[lch[rt]] + sum[rch[rt]];
}
int qL, qR;
int query(int rt, int pre, int L, int R) {
int ans = 0;
if(qL <= L && R <= qR) return sum[rt] - sum[pre];
int M = (L + R) / 2;
if(qL <= M) ans += query(lch[rt], lch[pre], L, M);
if(qR > M) ans += query(rch[rt], rch[pre], M+1, R);
return ans;
}
//fail tree
vector<int> G[maxn];
int l[maxn], r[maxn], dfs_clock;
void dfs(int u) {
l[u] = ++dfs_clock;
for(int v : G[u]) dfs(v);
r[u] = dfs_clock;
}
//Trie
int sz;
int ch[maxn][26], f[maxn], pos[maxn], fa[maxn];
void insert(int id, char* s) {
int u = 0;
for(int i = 0; s[i]; i++) {
int c = s[i] - 'a';
if(!ch[u][c]) {
ch[u][c] = ++sz;
memset(ch[sz], 0, sizeof(ch[0]));
}
fa[ch[u][c]] = u;
u = ch[u][c];
}
pos[id] = u;
}
void getFail() {
queue<int> Q;
REP(c, 0, 26) if(ch[0][c]) {
Q.push(ch[0][c]);
G[0].PB(ch[0][c]);
}
while(!Q.empty()) {
int r = Q.front(); Q.pop();
REP(c, 0, 26) {
int u = ch[r][c];
if(!u) ch[r][c] = ch[f[r]][c];
else {
f[u] = ch[f[r]][c];
G[f[u]].PB(u);
Q.push(u);
}
}
}
}
int n, q;
char s[maxn];
int main() {
scanf("%d%d", &n, &q);
REP(i, 0, n) {
scanf("%s", s);
insert(i + 1, s);
}
getFail();
dfs(0);
int cur = 0, s;
REP(i, 1, n + 1) {
for(int j = pos[i]; j; j = fa[j]) {
cur++;
update(root[cur], root[cur-1], 1, dfs_clock, l[j]);
}
p[i] = root[cur];
}
while(q--) {
int x, y, k; scanf("%d%d%d", &x, &y, &k);
qL = l[pos[k]], qR = r[pos[k]];
printf("%d\n", query(p[y], p[x-1], 1, dfs_clock));
}
return 0;
}
CodeForces 547E Mike and Friends AC自动机 主席树的更多相关文章
- NOI 2011 阿狸的打字机(AC自动机+主席树)
题意 https://loj.ac/problem/2444 思路 多串匹配,考虑 \(\text{AC}\) 自动机.模拟打字的过程,先建出一棵 \(\text{Trie}\) 树,把它变成自动机 ...
- codeforces 547E Mike and Friends
codeforces 547E Mike and Friends 题意 题解 代码 #include<bits/stdc++.h> using namespace std; #define ...
- hdu 4117 GRE Words (ac自动机 线段树 dp)
参考:http://blog.csdn.net/no__stop/article/details/12287843 此题利用了ac自动机fail树的性质,fail指针建立为树,表示父节点是孩子节点的后 ...
- hdu 4117 -- GRE Words (AC自动机+线段树)
题目链接 problem Recently George is preparing for the Graduate Record Examinations (GRE for short). Obvi ...
- 【BZOJ2434】阿狸的打字机(AC自动机,树状数组)
[BZOJ2434]阿狸的打字机(AC自动机,树状数组) 先写个暴力: 每次打印出字符串后,就插入到\(Trie\)树中 搞完后直接搭\(AC\)自动机 看一看匹配是怎么样的: 每次沿着\(AC\)自 ...
- 【BZOJ2434】【NOI2011】阿狸的打字机(AC自动机,树状数组)
[BZOJ2434]阿狸的打字机(AC自动机,树状数组) 先写个暴力: 每次打印出字符串后,就插入到\(Trie\)树中 搞完后直接搭\(AC\)自动机 看一看匹配是怎么样的: 每次沿着\(AC\)自 ...
- Codeforces 547E - Mike and Friends(AC 自动机+树状数组)
题面传送门 好久每做过 AC 自动机的题了--做几个题回忆一下罢 AC 自动机能够解决多串匹配问题,注意是匹配,碰到前后缀的问题那多半不在 AC 自动机能解决的范围内. 在初学 AC 自动机的时候相信 ...
- Codeforces 86C Genetic engineering(AC自动机+DP)
题目大概是给几个DNA片段,求构造一个长度n的字符串的方案数,要求这个字符串每个位置的字符都属于某个包含于此字符串的DNA片段. 把那些DNA片段建一个AC自动机.考虑状态的表示: dp[len][x ...
- Codeforces 291 E Tree-String Problem AC自动机
Tree-String Problem 网上的dfs + kmp 复杂度就是错的, 除非算出根据下一个字符直接转移Next数组直接转移, 而求出Next[ i ][ 26 ]数组和丢进AC自动机里面没 ...
随机推荐
- ajax异步请求的使用情景
先上代码 var sysAllowedExts; $.ajax({ url: url+"/getExtSetting", //请求的url地址 dataType: "js ...
- PAT1137
题意 一个学生的成绩由上机,期中,期末共3部分构成,现要求找出有资格获得证书的同学们. 证书获得者要求:上机分至少200,最终成绩及格. 最终成绩的生成规则:若期中分>期末分,则f = 期中 * ...
- April 15 2017 Week 15 Saturday
Attitude is a little thing that makes a big difference. 小态度,大不同. Attitudes can make a big difference ...
- Torch.no_grad()影响MSE损失
相关描述 https://discuss.pytorch.org/t/torch-no-grad-affecting-outputs-loss/28595/3 今天在训练网络的时候,发现mseloss ...
- 如何获得C4C里某个code字段对应的描述信息
通过我这篇文章介绍的方法使用C4C OData服务去取服务订单数据(Sales Order): 如何用代码的方式取出SAP C4C销售订单创建后所有业务伙伴的数据 https://www.jiansh ...
- TCP与虚连接
http://bbs.csdn.net/topics/390262738 在TCP通信时,会建立一个从源端到目的端的虚拟连接.感觉这种连接类似电路交换,只是这种连接是虚拟存在的.发送的报文都应该是沿着 ...
- 2017.10.24 Java 详解 JVM 工作原理和流程
JVM工作原理和特点主要是指操作系统装入JVM是通过jdk中Java.exe来完成,通过下面4步来完成JVM环境. 1.创建JVM装载环境和配置 2.装载JVM.dll 3.初始化JVM.dll并挂界 ...
- R 语言学习日志 1
1. CSV文件的的读取与写出 2. 数据集筛选 3. 简单随机抽样 sample函数 正文: 1. CSV文件的的读取与写出 文件读取: df2 <- read.table(" ...
- ubuntu 报错E: Could not get lock /var/lib/dpkg/lock-frontend - open (11: Resource temporarily unav E: Unable to acquire the dpkg frontend lock (/var/lib/dpkg/lock-frontend), is another process us
1.配置xshell,查看虚拟机中ubuntu中网络ip ifconfig 报错 Command 'ifconfig' not found, but can be installed with: su ...
- javascript中的作用域与作用域链
前几天,在写一段js代码时,出现了一些问题,调了很长时间也没有调通,其原因是,我在处理变量的作用域时错误地沿用了C++的作用域机制.因此我回炉了一次. 如果你使用过C++或java等一系列的面向对象的 ...