AC自动机+树链剖分+线段树/树状数组+dfs序+树链的并

题意:给出n个母串和q个询问串,对于每个询问串输出有多少个母串包含这个询问串 N=∑|母串|<=10^5 Q=∑|询问串|<=3.6*10^5

由于是母串包含询问串,那么我们就对询问串建自动机,然后用母串在上面跑,跑一次的复杂度是母串的长度(不确定)

利用fail树的性质求解,每次母串跑到一个节点,就把这个节点到根的路径都加上1,说明母串当前匹配单词的一段后缀包含某个前缀,也就是母串包含某个前缀

由于是求出现次数,那么询问串多次出现在母串里只算一次,所以之前到根的路径上加重了,所以我们利用树链的并去重,将节点按dfs序排序,相邻两个节点的lca到根的路径-1,这样就使得每个节点到根的路径都等于1

然后就真写了一个树链剖分+单点查询,其实直接查子树就行了,直接在节点上打上+1标记,lca上打上-1标记,树状数组维护子树和就行了

#include<bits/stdc++.h>
using namespace std;
const int N = ;
int n, q;
vector<char> v[N];
char s[N];
struct ac_automation {
int root, cnt, tot, dfs_clock;
int child[N][], fail[N], top[N], fa[N], son[N], dep[N], size[N], dfn[N], mir[N], tree[N << ], pos[N], tag[N << ];
vector<int> G[N], p;
void insert(char s[], int id)
{
int len = strlen(s), now = root;
for(int i = ; i < len; ++i)
{
int t = s[i] - 'a';
if(child[now][t] == ) child[now][t] = ++cnt;
now = child[now][t];
}
pos[id] = now;
}
void build_fail()
{
queue<int> q;
for(int i = ; i < ; ++i) if(child[root][i])
{
q.push(child[root][i]);
G[root].push_back(child[root][i]);
}
while(!q.empty())
{
int u = q.front();
q.pop();
for(int i = ; i < ; ++i)
{
int &v = child[u][i];
if(v == ) v = child[fail[u]][i];
else
{
fail[v] = child[fail[u]][i];
q.push(v);
G[child[fail[u]][i]].push_back(v);
}
}
}
}
int lca(int u, int v)
{
while(top[u] != top[v])
{
if(dep[top[u]] < dep[top[v]]) swap(u, v);
u = fa[top[u]];
}
return dep[u] < dep[v] ? u : v;
}
void dfs(int u)
{
size[u] = ;
for(int i = ; i < G[u].size(); ++i)
{
int v = G[u][i];
dep[v] = dep[u] + ;
fa[v] = u;
dfs(v);
size[u] += size[v];
if(size[v] >= size[son[u]]) son[u] = v;
}
}
void dfs(int u, int acs)
{
dfn[u] = ++dfs_clock;
mir[dfn[u]] = u;
top[u] = acs;
if(son[u]) dfs(son[u], acs);
for(int i = ; i < G[u].size(); ++i)
{
int v = G[u][i];
if(v == son[u]) continue;
dfs(v, v);
}
}
void pushdown(int x, int l, int r)
{
if(tag[x] == ) return;
int mid = (l + r) >> ;
tag[x << ] += tag[x];
tag[x << | ] += tag[x];
tree[x << ] += tag[x] * (mid - l + );
tree[x << | ] += tag[x] * (r - mid);
tag[x] = ;
}
int query(int l, int r, int x, int pos)
{
if(l == r) return tree[x];
pushdown(x, l, r);
int mid = (l + r) >> ;
if(pos <= mid) return query(l, mid, x << , pos);
else return query(mid + , r, x << | , pos);
}
void update(int l, int r, int x, int a, int b, int delta)
{
if(l > b || r < a) return;
if(l >= a && r <= b)
{
tag[x] += delta;
tree[x] += (r - l + ) * delta;
return;
}
pushdown(x, l, r);
int mid = (l + r) >> ;
update(l, mid, x << , a, b, delta);
update(mid + , r, x << | , a, b, delta);
tree[x] = tree[x << ] + tree[x << | ];
}
void change(int u, int delta)
{
while(top[u])
{
update(, cnt + , , dfn[top[u]], dfn[u], delta);
u = fa[top[u]];
}
update(, cnt + , , , dfn[u], delta);
}
void put_string(int id)
{
int len = v[id].size(), now = root;
p.clear();
for(int i = ; i < len; ++i)
{
now = child[now][v[id][i] - 'a'];
p.push_back(dfn[now]);
}
sort(p.begin(), p.end());
p.erase(unique(p.begin(), p.end()), p.end());
for(int i = ; i < p.size(); ++i)
{
int u = p[i];
change(mir[u], );
}
for(int i = ; i < p.size(); ++i)
{
int u = p[i], v = p[i - ];
change(lca(mir[u], mir[v]), -);
}
}
int ask(int id)
{
return query(, cnt + , , dfn[pos[id]]);
}
} ac;
int main()
{
scanf("%d%d", &n, &q);
for(int i = ; i <= n; ++i)
{
scanf("%s", s);
int len = strlen(s);
for(int j = ; j < len; ++j) v[i].push_back(s[j]);
}
for(int i = ; i <= q; ++i)
{
scanf("%s", s);
ac.insert(s, i);
}
ac.build_fail();
ac.dfs();
ac.dfs(, );
for(int i = ; i <= n; ++i) ac.put_string(i);
for(int i = ; i <= q; ++i) printf("%d\n", ac.ask(i));
return ;
}

bzoj2780的更多相关文章

  1. 【bzoj2780】 Sevenk Love Oimaster

    http://www.lydsy.com/JudgeOnline/problem.php?id=2780 (题目链接) 题意 给出很多主串和很多询问串,求一个询问串在多少主串中出现过 Solution ...

  2. 【BZOJ2780】【SPOJ】Sevenk Love Oimaster(后缀自动机)

    [BZOJ2780][SPOJ]Sevenk Love Oimaster(后缀自动机) 题面 BZOJ 洛谷 题解 裸的广义后缀自动机??? 建立广义后缀自动机建立出来之后算一下每个节点被几个串给包括 ...

  3. 【BZOJ2780】[Spoj]8093 Sevenk Love Oimaster 广义后缀自动机

    [BZOJ2780][Spoj]8093 Sevenk Love Oimaster Description Oimaster and sevenk love each other.     But r ...

  4. [bzoj2780][Spoj8093]Sevenk Love Oimaster_广义后缀自动机

    Sevenk Love Oimaster bzoj-2780 Spoj-8093 题目大意:给定$n$个大串和$m$次询问,每次给出一个字符串$s$询问在多少个大串中出现过. 注释:$1\le n\l ...

  5. BZOJ2780(广义后缀自动机,set启发式合并)

    BZOJ2780(广义后缀自动机,set启发式合并) 题面 自己找去 HINT 就是给多个文本串,然后每次查询的时候问你这个串在多少个文本串中出现过.因为多个文本串,那么直接就往广义后缀自动机上思考啊 ...

  6. BZOJ2780——[Spoj]8093 Sevenk Love Oimaster

    0.题意:给定N个原始字符串S,M次查询某个特殊的字符串S'在多少个原始串中出现过. 1.分析:这个题我们第一感觉就是可以用后缀自动机来搞,然后我们发现不是本质不同的字串..求出现过的次数,也就是说多 ...

  7. Bzoj2780: [Spoj]8093 Sevenk Love Oimaster

    题目 传送门 Sol 就是广义\(sam\) 然后记录下每个状态属于哪些串,开\(set\)维护 \(parent\)树上启发式合并一下就好了 # include <bits/stdc++.h& ...

  8. BZOJ2780:[SPOJ8093]Sevenk Love Oimaster(广义SAM)

    Description Oimaster and sevenk love each other. But recently,sevenk heard that a girl named ChuYuXu ...

  9. 【BZOJ2780】Sevenk Love Oimaster【广义后缀自动机】

    题意 给出你n个字符串和q个查询,每个查询给出一个字符串s,对于每个查询你都要输出这个字符串s在上面多少个字符串中出现过. 分析 广义后缀自动机的裸题.建好SAM以后再跑一遍得到每个状态的ocu和la ...

  10. [BZOJ2780][SPOJ8093]Sevenk Love Oimaster

    bzoj luogu 题面 给定n个模板串,以及m个查询串. 依次查询每一个查询串是多少个模板串的子串. sol 广义后缀自动机裸题? 先建出\(SAM\),然后记录一下每个节点分别在多少个模板串里出 ...

随机推荐

  1. jenkins自动部署测试环境

    构建脚本如下: echo "当前目录":$(pwd)echo "当前时间":$(date +%Y-%m-%d_%H:%M)find ./ -type f -na ...

  2. Spring资源访问接口Resource

    该接口拥有对不同资源类型的实现类 boolean exists() 资源是否存在 boolean isOpen() 资源是否打开 URL getURL() 如果底层资源可以表示成URL,则该方法返回对 ...

  3. Bookshelf 2(poj3628,01背包,dp递推)

    题目链接:Bookshelf 2(点击进入) 题目解读: 给n头牛,给出每个牛的高度h[i],给出一个书架的高度b(所有牛的高度相加>书架高度b),现在把一些牛叠起来(每头牛只能用一次,但不同的 ...

  4. mybatis中resultMap引发的吐血bug

    简单的讲: 问题背景:如果在写mybatis中的resultMap时,不下心将resultMapde id写成映射接口的名字,会发生什么? 结论:单元测试进度条卡住但不报错, Tomcat运行不报错, ...

  5. acm学习指引

    acm学习心得及书籍推荐   一般要做到50行以内的程序不用调试.100行以内的二分钟内调试成功.acm主要是考算法的,主要时间是花在思考算法上,不是花在写程序与debug上. 下面给个计划练练: 第 ...

  6. 题解 NOI2018 归程

    题解 NOI2018 归程 题意 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 n 个节点.m 条边的无向连通图(节点的编号从 1 至 n).我们依次用 l, ...

  7. HDU - 6158 The Designer

    传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6158 本题是一个计算几何题——四圆相切. 平面上的一对内切圆,半径分别为R和r.现在这一对内切圆之间,按 ...

  8. flask——CSRFToken保护

    根据 csrf_token 校验原理,具体操作步骤有以下几步: 1.后端生成 csrf_token 的值,在前端请求登录或者注册界面的时候将值传给前端,传给前端的方式可能有以下两种: 在模板中的 Fr ...

  9. [luoguP1076] 寻宝(模拟)

    传送门 模拟就好! 然后需要把一圈的有楼梯的都记录一下,取膜乱搞. 代码 #include <cstdio> #include <iostream> #define N 100 ...

  10. noip模拟赛 街灯

    分析:对于前30%的数据直接暴力模拟即可,对于另外30%的数据,因为每次的p是一样的,所以可以用莫队来维护,先离散化一下,再用一个桶统计次数. 100%的做法和之前做过的一道模拟赛题很像,当模数很小的 ...