The best programmers of Embezzland compete to develop a part of the project called "e-Government" — the system of automated statistic collecting and press analysis.

We know that any of the k citizens can become a member of the Embezzland government. The citizens' surnames are a1, a2, ..., ak. All surnames are different. Initially all k citizens from this list are members of the government. The system should support the following options:

  • Include citizen ai to the government.
  • Exclude citizen ai from the government.
  • Given a newspaper article text, calculate how politicized it is. To do this, for every active government member the system counts the number of times his surname occurs in the text as a substring. All occurrences are taken into consideration, including the intersecting ones. The degree of politicization of a text is defined as the sum of these values for all active government members.

Implement this system.

Input

The first line contains space-separated integers n and k (1 ≤ n, k ≤ 105) — the number of queries to the system and the number of potential government members.

Next k lines contain the surnames a1, a2, ..., ak, one per line. All surnames are pairwise different.

Next n lines contain queries to the system, one per line. Each query consists of a character that determines an operation and the operation argument, written consecutively without a space.

Operation "include in the government" corresponds to the character "+", operation "exclude" corresponds to "-". An argument of those operations is an integer between 1 and k — the index of the citizen involved in the operation. Any citizen can be included and excluded from the government an arbitrary number of times in any order. Including in the government a citizen who is already there or excluding the citizen who isn't there changes nothing.

The operation "calculate politicization" corresponds to character "?". Its argument is a text.

All strings — surnames and texts — are non-empty sequences of lowercase Latin letters. The total length of all surnames doesn't exceed 106, the total length of all texts doesn't exceed 106.

Output

For any "calculate politicization" operation print on a separate line the degree of the politicization of the given text. Print nothing for other operations.

Examples

Input
  1. 7 3
    a
    aa
    ab
    ?aaab
    -2
    ?aaab
    -3
    ?aaab
    +2
    ?aabbaa
Output
  1. 6
    4
    3
    6

题意:有M个不同的单词,和N个操作。先给出M个单词,然后N操作,

操作1,删去第i个单词(如果已经删了,则忽略);

操作2,添加,亦然。

操作3,给出字符串S,查询当前存在的单词在字符串S种出现了多少次(可以重复统计)。

思路:对M个单词建立AC自动机,然后是fail树,对fail树求dfs序。

假设没有求fail树和dfs序,只有fail指针,我求S出现次数的时候,S在AC自动机上跑,对于每一个当前Si在AC自动机的Now位置,都向上累加个数,表示以i为结尾的字符串,出现了多少次。

建立了fail树后,x的fail指针是x的爸爸,那么fail出现的时候,x也出现。即x出现的时候,子树都会++;所以在树状数组上+1,-1;

得到dfs序,询问串S时,在AC自动机上面跑,累加树状数组的贡献。

准确性:因为在询问串AC自动机上面跑的时候,我跑的深度是最大的,对它有贡献的都利用fail树和数状数组更新了,做到了不重不漏。

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. const int maxn=;
  4. int ch[maxn][],cnt=; //trie树
  5. int pos[maxn],st[maxn]; //在trie树的位置。
  6. int Laxt[maxn],Next[maxn],To[maxn],tot; //fail树
  7. int q[maxn],fail[maxn],head,tail; //fail树
  8. int in[maxn],out[maxn],sum[maxn],times;//dfs序
  9. char c[maxn];
  10. char getopt() { char c=getchar(); while(c!='+'&&c!='-'&&c!='?') c=getchar(); return c;}
  11. void addedge(int u,int v){ Next[++tot]=Laxt[u]; Laxt[u]=tot; To[tot]=v; }
  12. int insert()
  13. {
  14. int L=strlen(c+),Now=;
  15. for(int i=;i<=L;i++){
  16. if(!ch[Now][c[i]-'a']) ch[Now][c[i]-'a']=++cnt;
  17. Now=ch[Now][c[i]-'a'];
  18. } return Now;
  19. }
  20. void buildfail()
  21. {
  22. for(int i=;i<;i++){
  23. if(ch[][i]) q[++head]=ch[][i],fail[ch[][i]]=;
  24. else ch[][i]=;
  25. }
  26. while(tail<head){
  27. int Now=q[++tail];
  28. for(int i=;i<;i++){
  29. if(ch[Now][i]) {
  30. q[++head]=ch[Now][i]; fail[ch[Now][i]]=ch[fail[Now]][i];
  31. }
  32. else ch[Now][i]=ch[fail[Now]][i];
  33. }
  34. }
  35. for(int i=;i<=cnt;i++) addedge(fail[i],i);
  36. }
  37. void dfs(int u)
  38. {
  39. in[u]=++times;
  40. for(int i=Laxt[u];i;i=Next[i]) dfs(To[i]);
  41. out[u]=times;
  42. }
  43. void addsum(int x,int val){ while(x<=times){ sum[x]+=val; x+=(-x)&x;}}
  44. int query(int x){ int res=;while(x){res+=sum[x];x-=(-x)&x;}return res;}
  45. void solve()
  46. {
  47. int L=strlen(c+),Now=,ans=;
  48. for(int i=;i<=L;i++){
  49. Now=ch[Now][c[i]-'a'];
  50. ans+=query(in[Now]);
  51. }
  52. printf("%d\n",ans);
  53. }
  54. int main()
  55. {
  56. int N,M,x,i,j;
  57. scanf("%d%d",&N,&M);
  58. for(i=;i<=M;i++){
  59. st[i]=;
  60. scanf("%s",c+);
  61. pos[i]=insert();
  62. }
  63. buildfail();
  64. dfs();
  65. for(i=;i<=M;i++){
  66. addsum(in[pos[i]],);
  67. addsum(out[pos[i]]+,-);
  68. }
  69. for(i=;i<=N;i++){
  70. char opt=getopt();
  71. if(opt=='?'){
  72. scanf("%s",c+);
  73. solve();
  74. }
  75. else{
  76. scanf("%d",&x);
  77. if(opt=='+'){
  78. if(st[x]==) continue;st[x]=;
  79. addsum(in[pos[x]],);
  80. addsum(out[pos[x]]+,-);
  81. }
  82. else {
  83. if(st[x]==) continue; st[x]=;
  84. addsum(in[pos[x]],-);
  85. addsum(out[pos[x]]+,);
  86. }
  87. }
  88. }
  89. return ;
  90. }

CodeForces -163E :e-Government (AC自动机+DFS序+树状数组)的更多相关文章

  1. BZOJ 2434: [Noi2011]阿狸的打字机( AC自动机 + DFS序 + 树状数组 )

    一个串a在b中出现, 那么a是b的某些前缀的后缀, 所以搞出AC自动机, 按fail反向建树, 然后查询(x, y)就是y的子树中有多少是x的前缀. 离线, 对AC自动机DFS一遍, 用dfs序+树状 ...

  2. NOI 2011 阿狸的打字机 (AC自动机+dfs序+树状数组)

    题目大意:略(太长了不好描述) 良心LOJ传送门 先对所有被打印的字符串建一颗Trie树 观察数据范围,并不能每次打印都从头到尾暴力建树,而是每遍历到一个字符就在Trie上插入这个字符,然后记录每次打 ...

  3. BZOJ_2434_[NOI2011]_阿狸的打字机_(AC自动机+dfs序+树状数组)

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=2434 给出\(n\)个字符串,\(m\)个询问,对于第\(i\)个询问,求第\(x_i\)个字 ...

  4. BZOJ2434[Noi2011]阿狸的打字机——AC自动机+dfs序+树状数组

    题目描述 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的: l 输入小 ...

  5. 【BZOJ2434】[NOI2011]阿狸的打字机 AC自动机+DFS序+树状数组

    [BZOJ2434][NOI2011]阿狸的打字机 Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P ...

  6. BZOJ2434: [NOI2011]阿狸的打字机(AC自动机+dfs序+树状数组)

    [NOI2011]阿狸的打字机 题目链接:https://www.luogu.org/problemnew/show/P2414 题目背景 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. ...

  7. BZOJ 2434 阿狸的打字机(ac自动机+dfs序+树状数组)

    题意 给你一些串,还有一些询问 问你第x个串在第y个串中出现了多少次 思路 对这些串建ac自动机 根据fail树的性质:若x节点是trie中root到t任意一个节点的fail树的祖先,那么x一定是y的 ...

  8. BZOJ_3881_[Coci2015]Divljak_AC自动机+dfs序+树状数组

    BZOJ_3881_[Coci2015]Divljak_AC自动机+dfs序+树状数组 Description Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是 ...

  9. Codeforces Round #381 (Div. 2) D dfs序+树状数组

    D. Alyona and a tree time limit per test 2 seconds memory limit per test 256 megabytes input standar ...

随机推荐

  1. Leet Code OJ 237. Delete Node in a Linked List [Difficulty: Easy]

    题目: Write a function to delete a node (except the tail) in a singly linked list, given only access t ...

  2. 修改登陆织梦后台的“DedeCMS 提示信息”

    修改方法: 在dedecms程序的include目录中找到文件common.func.php并对其进行编辑,把其中的“DedeCMS 提示信息”修改为自己想要的内容提示: 在dedecms程序的默认管 ...

  3. CI学习总结

    1.CI自定义配置文件: 如:config/test.php <?php $config['test']['good'] = array('aa','bb'); 在控制器中这样调用: <? ...

  4. 软件测试人员需要精通的开发语言(4)--- Java

    接下来说下,当下最火的语言 - Java.Java是一种可以撰写跨平台应用程序的面向对象的程序设计语言.Java 技术具有卓越的通用性.高效性.平台移植性和安全性,广泛应用于PC.数据中心.游戏控制台 ...

  5. S2S4H整合注意问题

    整合过程中出现问题记录: 1.The import javax.servlet.http.HttpServletRequest cannot be resolved 解决办法:在tomcat的lib目 ...

  6. 九度OJ 1151:位操作练习 (位操作)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:1679 解决:924 题目描述: 给出两个不大于65535的非负整数,判断其中一个的16位二进制表示形式,是否能由另一个的16位二进制表示形 ...

  7. pip3 Fatal error in launcher: Unable to create process using '"' [转]

    在新环境上安装python的时候又再次遇到了这个情况,这次留意了一下,发现原来的文章有错误的地方,所以来更新一下,应该能解决大部分的问题. 环境是win8,原来只安装了python2.7.后来因为要用 ...

  8. 图床QAQ

  9. 【题解】Sumdiv

    [题解]Sumdiv 传送门 根据组合的乘法原理,一个数的所有约数和 \[ sum=\prod_{i=1} \Sigma_j^{a_i} p_i^j \] 所以任务就变成了分解\(A\)的质因数,分解 ...

  10. 【题解】HNOI2013比赛

    [题解][P3230 HNOI2013]比赛 将得分的序列化成样例给的那种表格,发现一行和一列是同时确定的.这个表格之前是正方形的,后来长宽都减去一,还是正方形.问题形式是递归的.这就启示我们可以把这 ...