热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

[HDU4787]GREWordsRevenge解题报告

这是我之前博客里提到的一道AC自动机的练手题,但是要完成这道题,我之前博客里提到的东西还不够,这里总结一下这道题。这道题不是一般的裸的AC

  这是我之前博客里提到的一道AC自动机的练手题,但是要完成这道题,我之前博客里提到的东西还不够,这里总结一下这道题。

  这道题不是一般的裸的AC自动机,它的询问和插入是交叉出现的所以用我之前写的板子不大合适,这道题在构建自动机时不能改变原有的 Trie树 的结构,所以没有代表字符串的结点的不需要去改它的值,所以我在 build() 函数 处做了一些修改。在复杂度方面,如果是强上普通的AC自动机,最差会达到O(n^2),感觉不太好。我们这里可以用到平方分割的套路,搞大小两个自动机,每次插入都在小自动机上进行,当小自动机里的结点达到一定量(sqrt(n))时,就将两自动机合并,并将小自动机清空。合并的方式也很好理解,就将两个自动机的节点一一对应,若小自动机上的结点在大自动机上没有,就在大自动机上新建结点,并修改它的值。总复杂度大概是这样:O(n*sqrt(n)(小的)+O(sqrt(n)*n)(大的)=O(n*sqrt(n))。其实那个上界也不一定要严格的根号n,自己大概估计一个常数就差不多了。哦对,还有一点,我发现好多板子里在构建新结点时都打了一个 newnode() 函数,包括我自己的板子也是这么打的。这样打其实是很慢的,我在做这道题时就各种 T飞 ,后来把这个去掉搞成其他的方法就快了很多(虽然我也不知道为什么),下面的代码里会体现。

  祝大家切题愉快

#include
#include
#include
#include
#include
#include
#define maxn (511111)
#define N (6000000)
#define il inline
#define ll long long
#define RG register
using namespace std;
char s[N],ss[N];struct Trie{int son[maxn][2],fail[maxn],size,root;bool val[maxn]; //标记是否出现il void init(){size&#61;1; root&#61;0;memset(son,0,sizeof(son));memset(val,0,sizeof(val));memset(fail,0,sizeof(fail));}il int idx(char c){ //卡常专用return c-&#39;0&#39;;}il int insert(char *s){ //插入新单词RG int cur&#61;root;for(RG int i&#61;0;s[i];i&#43;&#43;){ //卡常小技巧&#xff0c;不要每次计算lenRG int id&#61;idx(s[i]);if( !son[cur][id] ) son[cur][id]&#61;size&#43;&#43;;cur&#61;son[cur][id];}val[cur]&#61;true;return cur;}il bool in(char *s){ //查询单词是否在自动机内RG int cur&#61;root;for(RG int i&#61;0;s[i];i&#43;&#43;){RG int id&#61;idx(s[i]);if( !son[cur][id] ) return false;cur&#61;son[cur][id];}return val[cur];}il void build(){ //构建自动机queueQ;for(RG int i&#61;0;i<2;i&#43;&#43;)if( son[root][i] ) //只加入队列&#xff0c;不用修改值Q.push(son[root][i]);while(!Q.empty()){RG int cur&#61;Q.front(); Q.pop();for(RG int i&#61;0;i<2;i&#43;&#43;){RG int Son&#61;son[cur][i];if(!Son) continue;Q.push(Son); //同上RG int f&#61;fail[cur];while(f && !son[f][i] ) f&#61;fail[f];fail[Son]&#61;son[f][i];}}}il int query(char *s){ //查询次数RG int cur&#61;root,ans&#61;0;for(RG int i&#61;0;s[i];i&#43;&#43;){RG int id&#61;idx(s[i]);while(cur && !son[cur][id] ) cur&#61;fail[cur];cur&#61;son[cur][id];RG int k&#61;cur;while(k){ans&#43;&#61;val[k];k&#61;fail[k];}}return ans;}
}small,big;il void dfs(RG int u,RG int v){ //用于合并for(RG int i&#61;0;i<2;i&#43;&#43;)if(small.son[v][i]){RG int cur2&#61;small.son[v][i];if( !big.son[u][i] ){memset(big.son[big.size],0,sizeof(big.son[big.size]));big.son[u][i]&#61;big.size&#43;&#43;; //建立新结点}RG int cur1&#61;big.son[u][i];big.val[cur1] |&#61;small.val[cur2];dfs(cur1,cur2);}
}il void join(){dfs(0,0);small.init();big.build();
}int main(){RG int Case,n;scanf("%d",&Case);for(RG int k&#61;1;k<&#61;Case;k&#43;&#43;){printf("Case #%d:\n",k);scanf("%d",&n);small.init(); big.init();RG int l&#61;0;while(n--){scanf("%s",s);RG int len&#61;strlen(s&#43;1);ss[0]&#61;s[0];for(RG int i&#61;0;i2333 ) join();}else{l&#61;small.query(ss&#43;1)&#43;big.query(ss&#43;1);printf("%d\n",l);}}}return 0;
}

  真是服了这鬼畜的缩进。。。真难看。。。

转:https://www.cnblogs.com/Hero-of-someone/p/7157408.html



推荐阅读
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文介绍了Codeforces Round #321 (Div. 2)比赛中的问题Kefa and Dishes,通过状压和spfa算法解决了这个问题。给定一个有向图,求在不超过m步的情况下,能获得的最大权值和。点不能重复走。文章详细介绍了问题的题意、解题思路和代码实现。 ... [详细]
  • CF:3D City Model(小思维)问题解析和代码实现
    本文通过解析CF:3D City Model问题,介绍了问题的背景和要求,并给出了相应的代码实现。该问题涉及到在一个矩形的网格上建造城市的情景,每个网格单元可以作为建筑的基础,建筑由多个立方体叠加而成。文章详细讲解了问题的解决思路,并给出了相应的代码实现供读者参考。 ... [详细]
  • Go Cobra命令行工具入门教程
    本文介绍了Go语言实现的命令行工具Cobra的基本概念、安装方法和入门实践。Cobra被广泛应用于各种项目中,如Kubernetes、Hugo和Github CLI等。通过使用Cobra,我们可以快速创建命令行工具,适用于写测试脚本和各种服务的Admin CLI。文章还通过一个简单的demo演示了Cobra的使用方法。 ... [详细]
  • 开发笔记:实验7的文件读写操作
    本文介绍了使用C++的ofstream和ifstream类进行文件读写操作的方法,包括创建文件、写入文件和读取文件的过程。同时还介绍了如何判断文件是否成功打开和关闭文件的方法。通过本文的学习,读者可以了解如何在C++中进行文件读写操作。 ... [详细]
  • 李逍遥寻找仙药的迷阵之旅
    本文讲述了少年李逍遥为了救治婶婶的病情,前往仙灵岛寻找仙药的故事。他需要穿越一个由M×N个方格组成的迷阵,有些方格内有怪物,有些方格是安全的。李逍遥需要避开有怪物的方格,并经过最少的方格,找到仙药。在寻找的过程中,他还会遇到神秘人物。本文提供了一个迷阵样例及李逍遥找到仙药的路线。 ... [详细]
  • 本文介绍了一个Java猜拳小游戏的代码,通过使用Scanner类获取用户输入的拳的数字,并随机生成计算机的拳,然后判断胜负。该游戏可以选择剪刀、石头、布三种拳,通过比较两者的拳来决定胜负。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 动态规划算法的基本步骤及最长递增子序列问题详解
    本文详细介绍了动态规划算法的基本步骤,包括划分阶段、选择状态、决策和状态转移方程,并以最长递增子序列问题为例进行了详细解析。动态规划算法的有效性依赖于问题本身所具有的最优子结构性质和子问题重叠性质。通过将子问题的解保存在一个表中,在以后尽可能多地利用这些子问题的解,从而提高算法的效率。 ... [详细]
  • 本文介绍了UVALive6575题目Odd and Even Zeroes的解法,使用了数位dp和找规律的方法。阶乘的定义和性质被介绍,并给出了一些例子。其中,部分阶乘的尾零个数为奇数,部分为偶数。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • C++中的三角函数计算及其应用
    本文介绍了C++中的三角函数的计算方法和应用,包括计算余弦、正弦、正切值以及反三角函数求对应的弧度制角度的示例代码。代码中使用了C++的数学库和命名空间,通过赋值和输出语句实现了三角函数的计算和结果显示。通过学习本文,读者可以了解到C++中三角函数的基本用法和应用场景。 ... [详细]
  • 本文介绍了一个题目的解法,通过二分答案来解决问题,但困难在于如何进行检查。文章提供了一种逃逸方式,通过移动最慢的宿管来锁门时跑到更居中的位置,从而使所有合格的寝室都居中。文章还提到可以分开判断两边的情况,并使用前缀和的方式来求出在任意时刻能够到达宿管即将锁门的寝室的人数。最后,文章提到可以改成O(n)的直接枚举来解决问题。 ... [详细]
  • 本文介绍了Java中Hashtable的clear()方法,该方法用于清除和移除指定Hashtable中的所有键。通过示例程序演示了clear()方法的使用。 ... [详细]
author-avatar
手机用户2502917001
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有