【题目链接】 http://www.lydsy.com/JudgeOnline/problem.php?id=3483

【题目大意】

  给出一些串,同时给出m对前缀后缀,询问有多少串满足给出的前缀后缀模式,
  题目要求强制在线

【题解】

  我们对于给出的每个字符串正着插入字典树A,倒着插入字典树B,
  对于一个前缀来说,在字典树A上得到的dfs序[st,en]就是所有的匹配串,
  同理,后缀在字典树B上dfs序[st,en]表示所有的后缀匹配串,
  同时满足两者的需提供二维查询,因此我们以Adfsn为版本,Bdfsn为下标建立可持久化线段树,
  查询版本间区间和的差值就是符合条件的答案。

【代码】

#include <cstdio>
#include <algorithm>
#include <cstring>
#include <vector>
using namespace std;
const int S=2000010,N=2010;
struct Trie{
int tot,dfn,st[S],en[S],son[S][26],ansL,ansR;
int Tr(char c){return c-'a';}
void Init(){
for(int i=0;i<=tot;i++)for(int j=0;j<26;j++)son[i][j]=0;
dfn=tot=0;
}
int insert(char *s){
int x=0;
for(int l=strlen(s),i=0;i<l;i++){
if(!son[x][Tr(s[i])])son[x][Tr(s[i])]=++tot;
x=son[x][Tr(s[i])];
}return x;
}
void dfs(int x){
st[x]=++dfn;
for(int i=0;i<26;i++)if(son[x][i])dfs(son[x][i]);
en[x]=dfn;
}
void ask(char *s){
int x=0;
for(int l=strlen(s),i=0;i<l;i++){
if(!son[x][Tr(s[i])]){ansL=ansR=0;return;}
x=son[x][Tr(s[i])];
}ansL=st[x],ansR=en[x];
}
}A;
struct RevTrie{
int tot,dfn,st[S],en[S],son[S][26],ansL,ansR;
int Tr(char c){return c-'a';}
void Init(){
for(int i=0;i<=tot;i++)for(int j=0;j<26;j++)son[i][j]=0;
dfn=tot=0;
}
int insert(char *s){
int x=0;
for(int l=strlen(s),i=l-1;i>=0;i--){
if(!son[x][Tr(s[i])])son[x][Tr(s[i])]=++tot;
x=son[x][Tr(s[i])];
}return x;
}
void dfs(int x){
st[x]=++dfn;
for(int i=0;i<26;i++)if(son[x][i])dfs(son[x][i]);
en[x]=dfn;
}
void ask(char *s){
int x=0;
for(int l=strlen(s),i=l-1;i>=0;i--){
if(!son[x][Tr(s[i])]){ansL=ansR=0;return;}
x=son[x][Tr(s[i])];
}ansL=st[x],ansR=en[x];
}
}B;
namespace Persistent_Segment_Tree{
int l[N*40],r[N*40],v[N*40],tot,root[S];
int change(int x,int a,int b,int c,int p){
int y=++tot;v[y]=v[x]+p;
if(a==b)return y;
int mid=(a+b)>>1;
if(c<=mid)l[y]=change(l[x],a,mid,c,p),r[y]=r[x];
else l[y]=l[x],r[y]=change(r[x],mid+1,b,c,p);
return y;
}
int query(int x,int a,int b,int c,int d){
if(!x)return 0;
if(c<=a&&b<=d)return v[x];
int mid=(a+b)>>1,res=0;
if(c<=mid)res+=query(l[x],a,mid,c,d);
if(d>mid)res+=query(r[x],mid+1,b,c,d);
return res;
}
};
int n,m,posa[N],posb[N];
char s[S];
vector<int> G[S];
int main(){
using namespace Persistent_Segment_Tree;
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf(" %s",s);
posa[i]=A.insert(s);
posb[i]=B.insert(s);
}
A.dfs(0); B.dfs(0);
for(int i=1;i<=n;i++)G[A.st[posa[i]]].push_back(B.st[posb[i]]);
for(int i=1;i<=A.dfn;i++){
root[i]=root[i-1];
for(int j=0;j<G[i].size();j++)root[i]=change(root[i],1,B.dfn,G[i][j],1);
}
int ans=0;
scanf("%d",&m);
while(m--){
scanf(" %s",s);
int len=strlen(s);
for(int i=0;i<len;i++)s[i]=(s[i]-'a'+ans)%26+'a';
A.ask(s);
scanf(" %s",s);
len=strlen(s);
for(int i=0;i<len;i++)s[i]=(s[i]-'a'+ans)%26+'a';
B.ask(s);
if(A.ansL&&B.ansL)ans=query(root[A.ansR],1,B.dfn,B.ansL,B.ansR)-query(root[A.ansL-1],1,B.dfn,B.ansL,B.ansR);
else ans=0;
printf("%d\n",ans);
}return 0;
}

BZOJ 3483 SGU505 Prefixes and suffixes(字典树+可持久化线段树)的更多相关文章

  1. 主席树||可持久化线段树+离散化 || 莫队+分块 ||BZOJ 3585: mex || Luogu P4137 Rmq Problem / mex

    题面:Rmq Problem / mex 题解: 先离散化,然后插一堆空白,大体就是如果(对于以a.data<b.data排序后的A)A[i-1].data+1!=A[i].data,则插一个空 ...

  2. [BZOJ 4771]七彩树(可持久化线段树+树上差分)

    [BZOJ 4771]七彩树(可持久化线段树+树上差分) 题面 给定一棵n个点的有根树,编号依次为1到n,其中1号点是根节点.每个节点都被染上了某一种颜色,其中第i个节点的颜色为c[i].如果c[i] ...

  3. [BZOJ 3207] 花神的嘲讽计划Ⅰ【Hash + 可持久化线段树】

    题目链接:BZOJ - 3207 题目分析 先使用Hash,把每个长度为 k 的序列转为一个整数,然后题目就转化为了询问某个区间内有没有整数 x . 这一步可以使用可持久化线段树来做,虽然感觉可以有更 ...

  4. BZOJ.4771.七彩树(可持久化线段树)

    BZOJ 考虑没有深度限制,对整棵子树询问怎么做. 对于同种颜色中DFS序相邻的两个点\(u,v\),在\(dfn[u],dfn[v]\)处分别\(+1\),\(dfn[LCA(u,v)]\)处\(- ...

  5. BZOJ.3218.a + b Problem(最小割ISAP 可持久化线段树优化建图)

    BZOJ UOJ 首先不考虑奇怪方格的限制,就是类似最大权闭合子图一样建图. 对于奇怪方格的影响,显然可以建一条边\((i\to x,p_i)\),然后由\(x\)向\(1\sim i-1\)中权值在 ...

  6. BZOJ - 2588 Spoj 10628. Count on a tree (可持久化线段树+LCA/树链剖分)

    题目链接 第一种方法,dfs序上建可持久化线段树,然后询问的时候把两点之间的所有树链扒出来做差. #include<bits/stdc++.h> using namespace std; ...

  7. BZOJ 3439 Kpm的MCpassword Trie树+可持久化线段树

    题目大意:给定n个字符串,对于每一个字符串求以这个字符串为后缀的字符串中第k小的编号 首先将字符串反转 那么就变成了对于每一个字符串求以这个字符串为前缀的字符串中第k小的编号 然后考虑对字符串排序 那 ...

  8. BZOJ 4771: 七彩树 可持久化线段树+树链的并

    这个思路挺有意思的 ~ 利用树链的并来保证每个颜色只贡献一次,然后用可持久化线段树维护 code: #include <set> #include <cstdio> #incl ...

  9. 归并树 划分树 可持久化线段树(主席树) 入门题 hdu 2665

    如果题目给出1e5的数据范围,,以前只会用n*log(n)的方法去想 今天学了一下两三种n*n*log(n)的数据结构 他们就是大名鼎鼎的 归并树 划分树 主席树,,,, 首先来说两个问题,,区间第k ...

随机推荐

  1. javaScript书写规范

    命名规范. 常量名    全部大写并单词间用下划线分隔    如:CSS_BTN_CLOSE.TXT_LOADING对象的属性或方法名    小驼峰式(little camel-case)    如: ...

  2. ubuntu16.04 源码方法安装tensorflow

    参考博客:http://blog.csdn.net/zhaoyu106/article/details/52793183/,http://blog.csdn.net/u010900574/articl ...

  3. marshmallow: 简化Python对象系列化

    转载:http://www.thinksaas.cn/topics/0/594/594368.html marshmallow -一个轻量级的库用于将复杂对象转成简单的Python数据类型.或从简单的 ...

  4. The data protection operation was unsuccessful. This may have been caused by not having the user profile loaded for the current thread's user context,

    在iis7.0布署网站后运行的错误,大致意思是:数据保护操作是不成功的.这可能是由于没有为当前线程的用户加载用户配置文件的导致 解决办法: 先为自己的网站新建一个应用程序池,然后新建的应用程序池上右键 ...

  5. [NOI2007]货币兑换 「CDQ分治实现斜率优化」

    首先每次买卖一定是在某天 $k$ 以当时的最大收入买入,再到第 $i$ 天卖出,那么易得方程: $$f_i = \max \{\frac{A_iRate_kf_k}{A_kRate_k + B_k} ...

  6. js如何查看元素类型

    <script type="text/javascript"> //定义变量temp var temp = Object.prototype.toString.appl ...

  7. Python中使用LMDB

    在python中使用lmdb linux中,可以使用指令pip install lmdb安装lmdb包. 生成一个空的lmdb数据库文件 # -*- coding: utf-8 -*- import ...

  8. ORA-12514: TNS:listener does not currently know of service …

    问题描述: 今天数据库查询时遇到问题,具体情形如下截图所示: 问题分析: 看错误明显是TNS监听有问题,要么配置错了,要么数据库没起来.但是当前数据库起来了,也能正常连接使用,因此 考虑被查询对象可能 ...

  9. emacs设置了单例模式后无法设定文件关联解决办法

    emacs设置单例模式的本质就是使用下列参数启动: C:\emacs-24.5\bin\emacsclientw.exe --no-wait --alternate-editor="C:\e ...

  10. Java 从多层嵌套中访问内部类的成员

    一个内部类被嵌套多少层并不重要--它能透明地访问所有它能嵌入的外围类的所有成员 //: innerclasses/MultiNestingAccess.java // Nested classes c ...