BZOJ 3881: [Coci2015]Divljak [AC自动机 树链的并]
3881: [Coci2015]Divljak
题意:添加新文本串,询问某个模式串在多少种文本串里出现过
模式串建AC自动机,考虑添加一个文本串,走到的节点记录下来求树链的并
方法是按dfs序排序去重,每个点+1,相邻点lca-1
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
typedef long long ll;
const int N=4e6+5;
inline int read(){
char c=getchar();int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
return x*f;
}
int n, Q, op, x;
char s[N];
struct edge {int v, ne;} e[N];
int cnt=1, h[N];
inline void ins(int u, int v) {//printf("ins %d %d\n",u,v);
e[++cnt]=(edge){v, h[u]}; h[u]=cnt;}
namespace ac{
struct meow{int ch[26], fail, val;} t[N];
int pos[N], sz;
void insert(char *s, int id) {
int len = strlen(s+1), u=0;
for(int i=1; i<=len; i++) {
int c=s[i]-'a';
if(!t[u].ch[c]) t[u].ch[c] = ++sz;
u=t[u].ch[c];
}
pos[id]=u;
}
int q[N], head, tail;
void build() {
head=tail=1;
for(int i=0; i<26; i++) if(t[0].ch[i]) q[tail++] = t[0].ch[i];
while(head != tail) {
int u=q[head++];
for(int i=0; i<26; i++) {
int &v = t[u].ch[i];
if(!v) v = t[t[u].fail].ch[i];
else t[v].fail = t[t[u].fail].ch[i], q[tail++]=v;
}
}
for(int i=1; i<=sz; i++) ins(t[i].fail+1, i+1);
}
}using ac::t; using ac::sz;
int L[N], R[N], dfc, mx[N], deep[N], size[N], fa[N], top[N];
void dfs(int u) {
size[u]=1;
for(int i=h[u];i;i=e[i].ne) {
int v=e[i].v;
fa[v]=u; deep[v]=deep[u]+1;
dfs(v);
size[u]+=size[v];
if(size[v] > size[mx[u]]) mx[u]=v;
}
}
void dfs(int u, int anc) {
L[u] = ++dfc;
top[u] = anc;
if(mx[u]) dfs(mx[u], anc);
for(int i=h[u];i;i=e[i].ne) if(e[i].v != mx[u]) dfs(e[i].v, e[i].v);
R[u] = dfc;
}
int lca(int x, int y) {
while(top[x] != top[y]) {
if(deep[top[x]] < deep[top[y]]) swap(x, y);
x = fa[top[x]];
}
return deep[x] < deep[y] ? x : y;
}
int c[N];
inline void add(int p, int d) { //printf("add %d %d\n",p,d);
if(p==0) puts("nooo");
for(; p<=sz+1; p+=p&-p) c[p]+=d;}
inline int sum(int p) { //printf("sum %d\n",p);
if(p==-1) puts("nooo");
int ans=0; for(; p; p-=p&-p) ans+=c[p]; return ans;}
int a[N];
inline bool cmp(int x, int y) {return L[x]<L[y];}
void solve(char *s) { //printf("solve %s\n",s+1);
int len=strlen(s+1), u=0, p=0;
for(int i=1; i<=len; i++) {
u=t[u].ch[s[i]-'a'];
a[++p]=u+1;
}
sort(a+1, a+1+p, cmp); p = unique(a+1, a+1+p) - a - 1;
for(int i=1; i<=p; i++) {
add(L[a[i]], 1);
if(i!=1) add(L[ lca(a[i], a[i-1]) ], -1);
}
}
int que(int x) { //printf("que %d\n",x);
return sum(R[x]) - sum(L[x]-1);
}
int main() {
freopen("in","r",stdin);
n=read();
for(int i=1; i<=n; i++) scanf("%s",s+1), ac::insert(s, i);
ac::build();
dfs(1); dfs(1, 1); //puts("hi");
//for(int i=1; i<=sz+1; i++) printf("LR %d %d %d\n", i, L[i], R[i]);
Q=read();
for(int i=1; i<=Q; i++) {
op=read();
if(op==1) scanf("%s",s+1), solve(s);
else printf("%d\n", que(ac::pos[read()]+1));
}
}
BZOJ 3881: [Coci2015]Divljak [AC自动机 树链的并]的更多相关文章
- 【bzoj3881】[Coci2015]Divljak AC自动机+树链的并+DFS序+树状数组
题目描述 Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是空的. 接下来会发生q个操作,操作有两种形式: “1 P”,Bob往自己的集合里添加了一个字符串P. ...
- bzoj 3881: [Coci2015]Divljak AC自动机
题目大意: http://www.lydsy.com/JudgeOnline/problem.php?id=3881 题解: 这道题我想出了三种做法,不过只有最后一种能过. 第一种: 首先我们把所有的 ...
- BZOJ 3881[COCI2015]Divljak (AC自动机+dfs序+lca+BIT)
显然是用AC自动机 先构建好AC自动机,当B中插入新的串时就在trie上跑,对于当前点,首先这个点所代表的串一定出现过,然后这个点指向的fail也一定出现过.那么我们把每个点fail当作父亲,建一棵f ...
- BZOJ3881[Coci2015]Divljak——AC自动机+树状数组+LCA+dfs序+树链的并
题目描述 Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是空的. 接下来会发生q个操作,操作有两种形式: “1 P”,Bob往自己的集合里添加了一个字符串P. ...
- BZOJ 3881: [Coci2015]Divljak
3881: [Coci2015]Divljak Time Limit: 20 Sec Memory Limit: 768 MBSubmit: 553 Solved: 176[Submit][Sta ...
- BZOJ 3881 [Coci2015]Divljak(AC自动机+树状数组)
建立AC自动机然后,加入一个串之后考虑这个串的贡献.我们把这个串扔到AC自动机里面跑.最后对经过每一个点到的这个点在fail树的根的路径上的点有1的贡献.求链的并,我们把这些点按DFS序排序,然后把每 ...
- BZOJ 3881 [COCI2015]Divljak (Trie图+Fail树+树链的并+树状数组维护dfs序)
题目大意: Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是空的. 接下来会发生q个操作,操作有两种形式: “1 P”,Bob往自己的集合里添加了一个字符串P. ...
- bzoj 3881 [Coci2015]Divljak fail树+树链的并
题目大意 Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是空的. 接下来会发生q个操作,操作有两种形式: "1 P",Bob往自己的集合里添 ...
- 2019牛客多校八 H. How Many Schemes (AC自动机,树链剖分)
大意: 给定树, 每条边有一个字符集合, 给定$m$个模式串, $q$个询问$(u,v)$, 对于路径$(u,v)$中的所有边, 每条边从对应字符集合中取一个字符, 得到一个串$s$, 求$s$至少包 ...
随机推荐
- JavaScript八张思维导图
JS基本概念 JS操作符 JS基本语句 JS数组用法 Date用法 JS字符串用法 JS编程风格 JS编程实践 不知不觉做前端已经五年多了,无论是从最初的jQuery还是现在火热的Angular,Vu ...
- Hive_UDF函数中集合对象初始化的注意事项
UDF函数中定义的集合对象何时初始化 udf函数放在sql中对某个字段进行处理,那么在底层会创建一个该类的对象,这个对象不断的去调用这个evaluate(...)方法,截图如下: 1.1 如果说对 ...
- 急!!ftp登录错误,提示 530 not logged in,连接失败 ,,是怎么回事啊
愤怒地青鸟 | 浏览 68533 次 发布于2017-04-11 00:44 最佳答案 核心提示:Ftp登陆不了是很经常碰到的事,很多人常常是不加分析就发贴询问.老实说,这样既浪费自己时间,又浪费别人 ...
- thinkphp3.2后台模块怎么添加(admin),直接复制Home?还是在入口文件生成?
1.都可以,复制home改下命名空间也行,在入口添加下参数自动生成也行 2ThinkPHP3.2后支持模块化开发,在Home目录的同级目录下创建一个新的文件夹,命名为Admin,或者就如你自己所说,直 ...
- 宝塔linux面板.txt
安装命令: yum -y install screen wget && screen -S bt wget -O install.sh http://103.224.251.79:58 ...
- Struts的session问题
问题描述: 在一个action中设置session之后,在jsp中得不到session的值或者在另一个action中得不到session的值. 解决方案: 1.不要把session设置成为静态的,同时 ...
- css FlexBox 弹性盒子常用方法总结
总结一下弹性盒子常用的方法,弹性盒子的功能强大,这次我做了兼容性的felxbox,虽然代码多了一点,但在项目时候可以直接复制过来用,同时在项目上线的时候,如果这时候弹性盒子出了兼容问题,那就可急了~ ...
- cmd 指令
dir 展示当前文件夹内的文件 mkdir 创建文件夹 c: 访问C盘 (d: f:) cd\ 返回主文件 cd .. 返回上一文件夹 cd test 打开文件夹 访问文件夹 rd 删除文件夹 ...
- [Qt Quick] qmlscene工具的使用
qmlscene是Qt 5提供的一个查看qml文件效果的工具.特点是不需要编译应用程序. qmlscene = qml + scene (场景) qmlscene.exe位于Qt的安装目录下 (类似/ ...
- SQLSERVER存储过程语法详解
CREATE PROC [ EDURE ] procedure_name [ ; number ] [ { @parameter data_type } [ VARYING ] [ = default ...