BZOJ3172: [Tjoi2013]单词

【传送门:BZOJ3172】


简要题意:

  给出n个单词,你可以理解为将这些单词变成一个个段落,然后求出每个单词在所有段落中出现的次数


题解(一):

  刚开始不是很懂题目,结果发现将所有单词看成一篇文章,每个单词看成一个段落就懂了

  由于某种unbelievable的原因,我刚好做了AC自动机的专题训练,看到这道题就秒想AC自动机

  将每个单词放进AC自动机里,每个点的s表示有多少个单词经过,然后在构建失败指针的时候,通过队列来更新s值,怎么更新呢,假设有一个i点,它的失败指针指向j,设sj为从根到j点所构成的字符串,si为从根到i点所构成的字符串,那么我们可以知道sj为si的后缀,也说明了si中有sj的出现,那么就将j点的s值加上i点的s值,达到更新且不重复的目的来求出答案


 

参考代码(一):

#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<cmath>
using namespace std;
struct node
{int s,c[27],fail;node(){s=fail=0;memset(c,-1,sizeof(c));}
}t[1100000];
int tot,ans,n;
char a[1100000];
int ed[210];
void bt(int k,int root)
{int x=root,len=strlen(a+1);for(int i=1;i<=len;i++){int y=a[i]-'a'+1;if(t[x].c[y]==-1){t[x].c[y]=++tot;}x=t[x].c[y];t[x].s++;}ed[k]=x;
}
int list[1100000];
void bfs()
{int x;int head=1,tail=1;list[1]=0;while(head<=tail){x=list[head];for(int i=1;i<=26;i++){int son=t[x].c[i];if(son==-1)continue;if(x==0) t[son].fail=0;else{int j=t[x].fail;while(j!=0&&t[j].c[i]==-1) j=t[j].fail;t[son].fail=max(t[j].c[i],0);int x=t[son].fail,y=son;}list[++tail]=son;}head++;}for(int i=tail;i>=1;i--) t[t[list[i]].fail].s+=t[list[i]].s;
}
int main()
{ans=tot=0;scanf("%d",&n);for(int i=1;i<=n;i++){scanf("%s",a+1);bt(i,0);}bfs();for(int i=1;i<=n;i++){printf("%d\n",t[ed[i]].s);}return 0;
}

 


 

题解(二):

    A了这道题之后才发现早在之前就A了(心酸)

  那我们就来搞一下第二种想法

  很容易就想到AC自动机,但是单单是AC自动机还不行

  这时就要用AC自动机的延伸——fail树来做(时常膜一膜算法,有益身体健康)

  fail树这个玩意就是将AC自动机中fail指针当成是一条边,然后建成一棵树

  由于trie树上的每个点相当于一个字符串,所以这棵fail树父亲节点是儿子节点所构成字符串的最长后缀

  这样子对于这道题而言,fail树简直就是神一般的存在QAQ

  首先将每个字符串放进trie树里面,并且每经过一个trie树里的点,就用s数组记录这个字符串出现的个数,每经过一次就加一,然后构造AC自动机,然后在构造AC自动机的过程中,对每个点的fail指针进行建边,然后用dfs从根往下遍历,把s数组从下往上累加

  然后在输入每个字符串的时候,记录每个字符串的最后一个字符在trie树上的编号,然后一个一个输出s[end[i]](end[i]表示第i个字符串最后一个字母在trie树上的编号)


参考代码(二):

#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<cmath>
using namespace std;
struct node
{int c[27],fail;node(){fail=0;memset(c,-1,sizeof(c));}
}t[1100000];
int tot,n;
char st[1100000];
int s[1100000];
int end[1100000];
void bt(int root,int z)
{int x=root,len=strlen(st+1);for(int i=1;i<=len;i++){int y=st[i]-'a'+1;if(t[x].c[y]==-1){t[x].c[y]=++tot;}x=t[x].c[y];s[x]++;}end[z]=x;
}
struct edge
{int x,y,next;
}a[1100000];int len,last[1100000];
void ins(int x,int y)
{len++;a[len].x=x;a[len].y=y;a[len].next=last[x];last[x]=len;
}
queue<int> q;
void bfs()
{int x;q.push(0);while(q.empty()==0){x=q.front();for(int i=1;i<=26;i++){int son=t[x].c[i];if(son==-1)continue;if(x==0) t[son].fail=0;else{int j=t[x].fail;while(j!=0&&t[j].c[i]==-1) j=t[j].fail;t[son].fail=max(t[j].c[i],0);}ins(t[son].fail,son);q.push(son);}q.pop();}
}
void dfs(int x)
{for(int k=last[x];k;k=a[k].next){int y=a[k].y;dfs(y);s[x]+=s[y];}
}
int main()
{tot=0;scanf("%d",&n);for(int i=1;i<=n;i++){scanf("%s",st+1);bt(0,i);}len=0;memset(last,0,sizeof(last));bfs();dfs(0);for(int i=1;i<=n;i++) printf("%d\n",s[end[i]]);return 0;
}

 

 

转载于:https://www.cnblogs.com/Never-mind/p/7628734.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/282232.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL5.6二进制软件包编译安装详解(三)

一、软件环境 [rootlocalhost ~]# uname -r 3.10.0-862.el7.x86_64 [rootlocalhost ~]# cat /etc/redhat-release CentOS Linux release 7.5.1804 (Core) 二、安装部署过程详解 MySQL安装3种方式&#xff1a;1>rpm包安装应用文件默认安装在/usr/local 目录下2>源码编译需…

Java反射学习总结五(Annotation(注解)-基础篇)

Annotation(注解)简单介绍&#xff1a; 注解大家印象最深刻的可能就是JUnit做单元測试,和各种框架里的使用了。本文主要简介一下注解的用法&#xff0c;下篇文章再深入的研究。 annotation并不直接影响代码语义。可是它可以被看作类似程序的工具或者类库。它会反过来对正在执行…

使用autok3s 安装k3s 集群 和 kuboard 管理集群

一、k3s介绍1.1 什么是k3s?k3s 是经过 CNCF 认证的由 Rancher 公司开发维护的一个轻量级的 Kubernetes 发行版&#xff0c;内核机制还是和 k8s 一样&#xff0c;但是剔除了很多外部依赖以及 K8s 的 alpha、beta 特性&#xff0c;同时改变了部署方式和运行方式&#xff0c;目的…

Nginx—— Rewrite规则的使用

一、使用场景 1、URL访问跳转 &#xff08;1&#xff09;页面跳转 &#xff08;2&#xff09;兼容性支持&#xff08;比如新老版本交替时&#xff0c;给老版本一条访问道路&#xff09; &#xff08;3&#xff09;展示效果&#xff08;比如缩短前台界面的地址栏的url&#…

java对象实例化的方式

java对象实例化的方式有以下几种&#xff1a;1、使用new2、工厂模式3、反射4、clone()方法5、反序列化方式 /** 实现Cloneable和Serializable接口 */public class Book implements Cloneable, Serializable {private static final long serialVersionUID 1L; private Integer …

iOS-生成二维码图片【附中间带有小图标二维码】(QRCode)

生成二维码图片也是项目中常用到的&#xff0c;二维码的扫描Git上有很多好用的&#xff0c;这里主要说下二维码的生成 1.普通二维码 方法 /**生成二维码QRStering&#xff1a;字符串imageFloat&#xff1a;二维码图片大小*/ (UIImage *)createQRCodeWithString:(NSString *)QRS…

libubox

lbubox是openwrt的一个核心库&#xff0c;封装了一系列基础实用功能&#xff0c;主要提供事件循环&#xff0c;二进制格式处理&#xff0c;linux链表实现和一些JSON辅助处理。 它的目的是以动态链接库方式来提供可重用的通用功能&#xff0c;给其他模块提供便利和避免再造轮子。…

社区纠纷不断:程序员何苦为难程序员

出品 | OSC开源社区&#xff08;ID&#xff1a;oschina2013)今年年初&#xff0c;我们报道“因为被多人侮辱大吼&#xff0c;Swift 之父正式退出 Swift 核心团队”。诸如此类的“语言暴力”、“网络暴力”事件在开源社区乃至整个 IT 社区屡见不鲜。多个技术社区&#xff0c;都出…

PHP 分布式集群中session共享问题以及session有效期的设置

一、Session的原理 以下以默认情况举例&#xff1a; session_start();之后&#xff0c;会生成一个唯一的session_id&#xff0c;每一个用户对应唯一一个session_id&#xff0c;每一个session_id对应服务器端的一个session文件。这个session文件存储着当前session_id的信息&am…

[SDOI2009]Bill的挑战——全网唯一 一篇容斥题解

全网唯一一篇容斥题解 Description Solution 看到这个题&#xff0c;大部分人想的是状压dp 但是我是个蒟蒻没想到&#xff0c;就用容斥切掉了。 并且复杂度比一般状压低&#xff0c; &#xff08;其实这个容斥的算法&#xff0c;提出来源于ywy_c_asm&#xff09; &#xff08;然…

[NOIP2015提高组]运输计划

题目&#xff1a;BZOJ4326、洛谷P2680、Vijos P1983、UOJ#150、codevs4632、codevs5440。 题目大意&#xff1a;有一棵带权树&#xff0c;有一些运输计划&#xff0c;第i个运输计划从ai到bi&#xff0c;耗时为ai到bi的距离&#xff0c;所有运输计划一起开始。现在可以把一条边权…

对象存储OSS服务

一、oss是什么 阿里云对象存储服务&#xff08;Object Storage Service&#xff0c;简称OSS&#xff09;为您提供基于网络的数据存取服务。使用OSS&#xff0c;您可以通过网络随时存储和调用包括文本、图片、音频和视频等在内的各种非结构化数据文件。 阿里云OSS将数据文件以…

《Access 2007开发指南(修订版)》一一1.5 什么是数据库对象

本节书摘来自异步社区出版社《Access 2007开发指南(修订版)》一书中的第1章&#xff0c;第1.5节&#xff0c;作者&#xff1a; 【美】Alison Balter&#xff0c;更多章节内容可以访问云栖社区“异步社区”公众号查看。 1.5 什么是数据库对象 Access 2007开发指南(修订版)正如前…

ETL工具kettle的组件--生成记录

今天介绍下kettle的一个比较实用的组件——生成记录&#xff1b;当我们想将一部分文本数据变成数据行&#xff0c;每个字段作为一个数据行的一个列&#xff0c;那么我们可以利用这个组件&#xff1b;它的位置在双击点开根据自己的实际需要进行设置当设置后&#xff0c;可以点击…

Linux学习笔记一

linux  kernel lib module shell tools ls -la&#xff1a; 显示所有文件包括隐藏文件  cat /proc/cpuinfo&#xff1a; 显示cpu信息 man man  /string&#xff1a; 向上搜索string字符串 继续按下小写n向上搜索  ?string&#xff1a; 向下搜索string字符串 继续按下大…

PHP中路由和rewrite的使用

一、场景介绍&#xff1a; 1、简化url地址&#xff0c;方便大家记忆 2、有利于搜索引擎优化 3、安全&#xff08;让用户看不出网站的目录结构&#xff09; 举例&#xff1a;比如我这里将main控制器中的bb方法路由到kk&#xff0c;这样&#xff0c;我们a标签请求跳转到cp.xi…

《NoSQL权威指南》导读

引言 NoSQL权威指南“没有什么会比引入新秩序更难&#xff0c;因为创新者必须要面对那些在旧环境中已经做得很好的对手&#xff0c;以及那些在新环境中做得很好的冷漠者。” ——Niccolo Machiavelli [1] 在过去的几十年&#xff0c;我已经通过Elsevier/Morgan Kaufmann出版社出…

zookeeper的单实例和伪集群部署

原文链接: http://gudaoyufu.com/?p1395 zookeeper工作方式 ZooKeeper 是一个开源的分布式协调服务&#xff0c;由雅虎创建&#xff0c;是 Google Chubby 的开源实现。 分布式应用程序可以基于 ZooKeeper 实现诸如数据发布/订阅、负载均衡、命名服务、分布式协 调/通知、集群管…

PHP开发常见功能实现流程

一、pc端网站登录 1、获取并过滤用户提交的用户名和密码以及验证码 2、验证用户提交验证码和session中的验证码是否一致 3、验证用户名是否存在 4、根据用户名获取密码&#xff0c;并校验密码是否一致 5、密码一致&#xff0c;则登录成功&#xff0c;跳转到对应的首页 图示…

七牛直播云服务技术揭秘

以下根据七牛云首席布道师何李石现场演讲内容整理。 直播模型及其实现 一个通用的直播模型一般包括三个模块&#xff1a;主播方、服务器端和播放端。 首先是主播方&#xff0c;它是产生视频流的源头&#xff0c;由一系列流程组成&#xff1a; 第一&#xff0c;通过一定的设备来…