博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
BZOJ3172:[TJOI2013]单词——题解
阅读量:5937 次
发布时间:2019-06-19

本文共 1214 字,大约阅读时间需要 4 分钟。

某人读论文,一篇论文是由许多单词组成。但他发现一个单词会在论文中出现很多次,现在想知道每个单词分别在论文中出现多少次。

fail树的棵题。

朴素想法就是建立AC自动机,每个串都匹配一遍,这样显然TLE。

但是实际上我们就是相当于查询一个单词x在另一个单词y内出现了多少次。

对于上面这个问题,一种朴素想法是AC自动机,把y单词的所有结点的fail都跑一遍,看看有没有指向x末尾的,有答案++。

其实我们只用到了fail指针,所以不如我们把fail提出来重新建树,减少状态。

令cnt[i]为0~i的路径拼起来字符串出现次数。

这样我们建立了fail树,从下往上累加cnt即可,最终答案就是查询单词的末尾记录的cnt。

我们因为只进行了一次dfs,比之前查询n次的效率显然高到不知道哪里去了。

#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;typedef long long ll;const int N=220;const int S=1e6+5;inline int read(){ int X=0,w=0;char ch=0; while(!isdigit(ch)){w|=ch=='-';ch=getchar();} while(isdigit(ch))X=(X<<3)+(X<<1)+(ch^48),ch=getchar(); return w?-X:X;}struct trie{ int a[26],fail;}tr[S];struct node{ int to,nxt;}e[S];int m,tot,cnt[S],ed[N],sum,head[S];char s[S];queue
q;inline void add(int u,int v){ e[++sum].to=v;e[sum].nxt=head[u];head[u]=sum;}void insert(int id){ int len=strlen(s),now=0; for(int i=0;i

+++++++++++++++++++++++++++++++++++++++++++

+本文作者:luyouqi233。               +

+欢迎访问我的博客: +

+++++++++++++++++++++++++++++++++++++++++++

转载于:https://www.cnblogs.com/luyouqi233/p/8997360.html

你可能感兴趣的文章
学习日志---树回归(回归树,模型树)
查看>>
Gene6_FTP_Server_高级配置
查看>>
centos 7编译安装nginx
查看>>
【学神】1-15 linux启动及常见故障的排除
查看>>
Android SDK 在线更新镜像服务器资源
查看>>
重新定义工作站的“边界”
查看>>
第三方推送已死
查看>>
回档|神经网络
查看>>
Apache Spark源码走读之12 -- Hive on Spark运行环境搭建
查看>>
阿里云跨服务器文件拷贝
查看>>
GetWindowRect
查看>>
6.<1>四则运算的研究[栈]
查看>>
java程序员笑不死的经历ส้้้้้้้้้
查看>>
php-fpm配置
查看>>
c++头文件和#include 学习笔记
查看>>
第四天(考试)
查看>>
关于VUE的路由地址问题
查看>>
node-buffer解读
查看>>
Vue 2.x折腾记 - (22) Vue 打包图片在safari不显示的问题
查看>>
ES6中的class
查看>>