HDU-4787 GRE Words Revenge 解题报告
这是我之前博客里提到的一道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<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<queue>
#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=1; root=0;
memset(son,0,sizeof(son));
memset(val,0,sizeof(val));
memset(fail,0,sizeof(fail));
}
il int idx(char c){ //卡常专用
return c-'0';
}
il int insert(char *s){ //插入新单词
RG int cur=root;
for(RG int i=0;s[i];i++){ //卡常小技巧,不要每次计算len
RG int id=idx(s[i]);
if( !son[cur][id] ) son[cur][id]=size++;
cur=son[cur][id];
}
val[cur]=true;
return cur;
}
il bool in(char *s){ //查询单词是否在自动机内
RG int cur=root;
for(RG int i=0;s[i];i++){
RG int id=idx(s[i]);
if( !son[cur][id] ) return false;
cur=son[cur][id];
}
return val[cur];
}
il void build(){ //构建自动机
queue<int>Q;
for(RG int i=0;i<2;i++)
if( son[root][i] ) //只加入队列,不用修改值
Q.push(son[root][i]);
while(!Q.empty()){
RG int cur=Q.front(); Q.pop();
for(RG int i=0;i<2;i++){
RG int Son=son[cur][i];
if(!Son) continue;
Q.push(Son); //同上
RG int f=fail[cur];
while(f && !son[f][i] ) f=fail[f];
fail[Son]=son[f][i];
}
}
}
il int query(char *s){ //查询次数
RG int cur=root,ans=0;
for(RG int i=0;s[i];i++){
RG int id=idx(s[i]);
while(cur && !son[cur][id] ) cur=fail[cur];
cur=son[cur][id];
RG int k=cur;
while(k){
ans+=val[k];
k=fail[k];
}
}
return ans;
}
}small,big;
il void dfs(RG int u,RG int v){ //用于合并
for(RG int i=0;i<2;i++)
if(small.son[v][i]){
RG int cur2=small.son[v][i];
if( !big.son[u][i] ){
memset(big.son[big.size],0,sizeof(big.son[big.size]));
big.son[u][i]=big.size++; //建立新结点
}
RG int cur1=big.son[u][i];
big.val[cur1] |=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=1;k<=Case;k++){
printf("Case #%d:\n",k);
scanf("%d",&n);
small.init(); big.init();
RG int l=0;
while(n--){
scanf("%s",s);
RG int len=strlen(s+1);
ss[0]=s[0];
for(RG int i=0;i<len;i++)
ss[i+1]=s[ (i+l%len+len)%len+1 ];
ss[len+1]='\0';
if(ss[0]=='+'){
if( small.in(ss+1) || big.in(ss+1) ) continue;
//若该单词已经存在就跳过
small.insert(ss+1); small.build();
if( small.size>2333 ) join();
}
else{
l=small.query(ss+1)+big.query(ss+1);
printf("%d\n",l);
}
}
}
return 0;
}
真是服了这鬼畜的缩进。。。真难看。。。
HDU-4787 GRE Words Revenge 解题报告的更多相关文章
- [HDU 4787] GRE Words Revenge (AC自动机)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4787 题目大意: 给你若干个单词,查询一篇文章里出现的单词数.. 就是被我水过去的...暴力重建AC自 ...
- ●HDU 4787 GRE Words Revenge
题链: http://acm.hdu.edu.cn/showproblem.php?pid=4787 题解: AC自动机(强制在线构造) 题目大意: 有两种操作, 一种为:+S,表示增加模式串S, 另 ...
- HDU 4787 GRE Words Revenge
Description Now Coach Pang is preparing for the Graduate Record Examinations as George did in 2011. ...
- [HDU4787]GRE Words Revenge 解题报告
这是我之前博客里提到的一道AC自动机的练手题,但是要完成这道题,我之前博客里提到的东西还不够,这里总结一下这道题. 这道题不是一般的裸的AC自动机,它的询问和插入是交叉出现的所以用我之前写的板子不大合 ...
- hdu 1556.Color the ball 解题报告
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1556 题目意思:有 n 个气球从左到右排成一排,编号依次为1,2,3,...,n.给出 n 对 a, ...
- hdu 1160 FatMouse's Speed 解题报告
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1160 题目意思:给出一堆老鼠,假设有 n 只(输入n条信息后Ctrl+Z).每只老鼠有对应的weigh ...
- hdu 1879 继续畅通工程 解题报告
题目链接:http://code.hdu.edu.cn/showproblem.php?pid=1879 这条题目我的做法与解决Constructing Roads的解法是相同的. 0 表示没有连通: ...
- hdu 1233 还是畅通工程 解题报告
题目链接:http://code.hdu.edu.cn/showproblem.php?pid=1233 并查集的运用, 实质就是求最小生成树.先对所有的村庄距离从小到大排序,然后判断村庄之间是否属于 ...
- hdu 1213 How Many Tables 解题报告
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1213 有关系(直接或间接均可)的人就坐在一张桌子,我们要统计的是最少需要的桌子数. 并查集的入门题,什 ...
随机推荐
- 关于通用的C#后台获取前台页面的标签的正则表达式
Regex reg = new Regex("<div[^>]*?class=\"类属性名称\"[^>]*>(.*?) </div> ...
- Akka(8): 分布式运算:Remoting-远程查找式
Akka是一种消息驱动运算模式,它实现跨JVM程序运算的方式是通过能跨JVM的消息系统来调动分布在不同JVM上ActorSystem中的Actor进行运算,前题是Akka的地址系统可以支持跨JVM定位 ...
- Python模块之ConfigParser - 读写配置文件
Python 标准库的 ConfigParser 模块提供一套 API 来读取和操作配置文件. 配置文件的格式 a) 配置文件中包含一个或多个 section, 每个 section 有自己的 opt ...
- 360安全检测出的WordPress漏洞的修复方法
1.跨站脚本攻击(XSS) 这个漏洞注意是因为用户评论可以提交代码,有安全风险.虽然你的WordPress以及是最新版,但是你的WordPress主题却不一定跟着更新!因此,需要稍微修改一下评论相关的 ...
- IDEA报错处理:Application Server was not connected before run configuration stop, reason: Unable to ping server at localhost:8080
把wildfly的整个软件包更换成新的,配置文件重新配置,JBOSS_HOME环境变量修改成新的,在wildfly-10.1.0.FinalForTest\modules\system\layers\ ...
- 【源码分享】mui实现简单的手机音乐播放器
mui实现简单的手机音乐播放器 最近先来无事,我用mui写了一个可以跨页面控制的音乐播放器.主要功能有上一曲,下一曲,播放,暂停,感兴趣的可以继续看下去. 说的总是不实在,直接上源码,有兴趣的可以读下 ...
- RPM基础知识
RPM包命名原则 httpd-2.2.15-15.el6.centos.1.i686.rpm httpd 软件包名 2.2.15 软件版本 15 软件发布的次数 el ...
- python核心数据结构之字典
 [TOC ...
- 【Android Developers Training】 7. 添加Action Buttons
注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...
- Jenkins+Ant+TestNG+Testlink自动化构建集成(完整版)
这段时间折腾自动化测试,之前都是在Eclipse工程里面手工执行自动化测试脚本,调用Testlink API执行测试用例,目前搭建Jenkins自动化构建测试的方式,实现持续构建,执行自动化测试. 硬 ...