HDU 6096 AC自动机
n个字符串 m个询问 每个询问给出前后缀 并且不重合 问有多少个满足
m挺大 如果在线只能考虑logn的算法
官方题解:对n个串分别存正序倒序 分别按照字典序sort 每一个串就可以被化作一个点 那么对于每一个询问 前后缀都会有一个范围 那就是求这个范围里面的点的个数
官方题解转化到了求矩形并 然而没理解怎么转化 倒是想出来了kdtree直接求范围点 由于前后缀不能重合 即如果有aa aa这个询问 我的办法会找到aaa 就要进行去重
我用map记录 然后每次询问都查询一下。。跑的特别慢。。
但是在写这个博客的时候想出来 可以存三维点 第三维代表这n个串的长度 那么就可以查询len>某个值 进行三维查询
不知道复杂度是不是还会很高。。明天试一下
我的做法:对于n个串 abcde -> abcde*abcde m个询问 ab de -> de*ab 这样直接离线n个串 先把m个询问插进去 记录一下id 直接用n个串跑 最后输出ans
在其中记录fail树上每个点的dep 如果这个点的dep > n个串的长度 就不对答案贡献
但是这样又有了一个新的问题 一模一样的询问会少算 于是我用map<string,int>来记录一下是否已经出现 如果出现了那么这个询问不insert 而是直接指向最早出现的询问
写了一万个bug...不过顺便学习了线段树求矩形面积/周长并
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<vector>
#include<map>
#include<queue>
#include<algorithm>
using namespace std;
#define L long long
#define pb push_back string a[100050] ;
int n , m ;
char s[500050 * 2] , s2[500050*2];
int ans[100050] ;
struct tire{
int nex[700010*2][27] , fail[700010*2], id[700010*2];
int last[700010*2] ;
int dep[700010*2] ;
int root,l;
int newnode(){
for(int i=0;i<27;i++)nex[l][i]=-1;
last[l]=0;
id[l++]=0;
return l-1;
}
void init(){
l=0;
root=newnode();
dep[root]=0;
}
void inse(string s ,int idd){
int le = s.length() ;
int now=root;
for(int i=0;i<le;i++){
int x;
if(s[i]>='a'&&s[i]<='z')x=s[i]-'a';
else x=26;
if(nex[now][x] == -1) {
nex[now][x] = newnode(); }
now=nex[now][x];
dep[now]=i+1;
}
id[now]=idd;
}
void build() {
queue<int>q;
fail[root]=root;
for(int i=0;i<27;i++){
if(nex[root][i]==-1)
nex[root][i]=root;
else {
fail[nex[root][i]]=root;
q.push(nex[root][i]);
}
}
while(!q.empty()){
int now=q.front();q.pop();
for(int i= 0;i<27;i++){
if(nex[now][i]==-1)
nex[now][i]=nex[fail[now]][i] ;
else {
fail[nex[now][i]] = nex[fail[now]][i] ;
q.push(nex[now][i]) ;
}
}
}
}
void query(string s, int cd , int num){
int le = s.length() ;
int now=root;
int res = 0 ;
for(int i = 0 ; i < le ; i ++ ) {
int x;
if(s[i]>='a'&&s[i]<='z')x=s[i]-'a';
else x=26;
now = nex[now][x];
int temp=now;
while(temp!=root){
if(id[temp]!=0 && dep[temp] <= cd+1){
ans[id[temp]]++;
}
temp = fail[temp] ;
}
}
}
}ac;
map<string,int>mp;
int main () {
int t ; scanf("%d",&t);
while(t -- ) {
mp.clear();
ac.init() ;
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++){
cin>>a[i];
}
string c;
memset(ans,0,sizeof(ans));
for(int i=1;i<=m;i++){
string s1,s2;
cin>>s1>>s2;
s2+="*";s2+=s1;
if(mp[s2]!=0)
ans[i]=mp[s2]*-1;
else{
ac.inse(s2,i);
mp[s2]=i;
}
}
ac.build();
for(int i=1;i<=n;i++){
int len=a[i].length() ;
string s=a[i];s+="*";s+=a[i];
ac.query(s , len , i) ;
}
for(int i=1;i<=m;i++){
if(ans[i]<0){
printf("%d\n",ans[ans[i]*(-1)]);
}
else {
printf("%d\n",ans[i]);
}
}
}
}
HDU 6096 AC自动机的更多相关文章
- hdu 2896 AC自动机
// hdu 2896 AC自动机 // // 题目大意: // // 给你n个短串,然后给你q串长字符串,要求每个长字符串中 // 是否出现短串,出现的短串各是什么 // // 解题思路: // / ...
- hdu 3065 AC自动机
// hdu 3065 AC自动机 // // 题目大意: // // 给你n个短串,然后给你一个长串,问:各个短串在长串中,出现了多少次 // // 解题思路: // // AC自动机,插入,构建, ...
- hdu 5880 AC自动机
Family View Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total ...
- hdu 2296 aC自动机+dp(得到价值最大的字符串)
Ring Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submis ...
- hdu 2825 aC自动机+状压dp
Wireless Password Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
- hdu 3065 AC自动机(各子串出现的次数)
病毒侵袭持续中 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Sub ...
- HDU 5384 AC自动机
链接:http://acm.hdu.edu.cn/showproblem.php?pid=5384 题意:给n个母串,给m个匹配串,求每个母串依次和匹配串匹配,能得到的数目和. 分析:之前并不知道AC ...
- HDU 2222 AC自动机模板题
题目: http://acm.hdu.edu.cn/showproblem.php?pid=2222 AC自动机模板题 我现在对AC自动机的理解还一般,就贴一下我参考学习的两篇博客的链接: http: ...
- HDU 2846 (AC自动机+多文本匹配)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2846 题目大意:有多个文本,多个模式串.问每个模式串中,有多少个文本?(匹配可重复) 解题思路: 传统 ...
随机推荐
- PHP实现对短信验证码发送次数的限制(防机刷验证码)
PHP实现对短信验证码发送限制(防止机刷验证码) 对用户获取短信验证码的手机号.ip.和浏览器(使用唯一标识)进行限制.本文介绍的方法是对用户每天只能通过同一浏览器或同一ip地址获取验证码10次或者同 ...
- Viewpage实现左右无限滑动
实现逻辑参考:http://www.cnblogs.com/xinye/archive/2013/06/09/3129140.html 代码:如下 public class MainActivity ...
- IIPP迷你项目(四)"Pong"
1 小球在墙面的反弹 1-1 让小球在画布上匀速运动 在这一步中,首先应该明确小球是如何匀速运动的.它的方法是规定一个列表v,Scott老师说这代表着“速度(Velocity)”,但是我觉得也可以拿“ ...
- sqlserver 脚本和批处理指令小结
一.脚本基础 1.USE语句 设置当前数据库. 2.声明变量 语法:DECLARE @变量名 变量类型 在声明变量后,给变量赋值之 ...
- CSS3选择器:nth-child与:nth-of-type区别
一.:nth-child 1.1 说明 :nth-child(n)选择器匹配属于其父元素的第N个子元素,不论元素的类型.n可以是数字.关键词或公式. 注意:如果第N个子元素与选择的元素类型不同则样式无 ...
- 我的Android进阶之旅------>Android中查看应用签名信息
一.查看自己的证书签名信息 如上一篇文章<我的Android进阶之旅------>Android中制作和查看自定义的Debug版本Android签名证书>地址:http://blog ...
- Thrift官方安装手册(译)
本篇是Thrift官网安装文档的翻译,原地址点击这里.Thrift之前是不支持Windows的.但是似乎0.9版本以后已经支持Window了.介绍了Thrift安装的环境要求以及在centos,Deb ...
- go语言之并发编程同步一
前面介绍了采用go语法的并行操作以及channel.既然是并行操作,那么就涉及到数据原子性以及同步的问题.所以在Go里面也需要采用同步的机制. 互斥锁: 由标准库代码包sync中的Mutex结构体类型 ...
- 005-搭建框架-实现AOP机制【二】AOP技术
一.什么是AOP aspect-oriented programming,面向切面编程,对oop的一种补充. 著名示例,aspectj,spring+aspectj 二.aop之代码重构 2.1.代 ...
- 剑指offer 面试36题
面试36题: 题:二叉搜索树与双向链表 题目:输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表.要求不能创建任何新的结点,只能调整树中结点指针的指向. 解题思路一:由于输入的一个二叉搜索树, ...