BZOJ 3483 SGU505 Prefixes and suffixes(字典树+可持久化线段树)
【题目链接】 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(字典树+可持久化线段树)的更多相关文章
- 主席树||可持久化线段树+离散化 || 莫队+分块 ||BZOJ 3585: mex || Luogu P4137 Rmq Problem / mex
题面:Rmq Problem / mex 题解: 先离散化,然后插一堆空白,大体就是如果(对于以a.data<b.data排序后的A)A[i-1].data+1!=A[i].data,则插一个空 ...
- [BZOJ 4771]七彩树(可持久化线段树+树上差分)
[BZOJ 4771]七彩树(可持久化线段树+树上差分) 题面 给定一棵n个点的有根树,编号依次为1到n,其中1号点是根节点.每个节点都被染上了某一种颜色,其中第i个节点的颜色为c[i].如果c[i] ...
- [BZOJ 3207] 花神的嘲讽计划Ⅰ【Hash + 可持久化线段树】
题目链接:BZOJ - 3207 题目分析 先使用Hash,把每个长度为 k 的序列转为一个整数,然后题目就转化为了询问某个区间内有没有整数 x . 这一步可以使用可持久化线段树来做,虽然感觉可以有更 ...
- BZOJ.4771.七彩树(可持久化线段树)
BZOJ 考虑没有深度限制,对整棵子树询问怎么做. 对于同种颜色中DFS序相邻的两个点\(u,v\),在\(dfn[u],dfn[v]\)处分别\(+1\),\(dfn[LCA(u,v)]\)处\(- ...
- BZOJ.3218.a + b Problem(最小割ISAP 可持久化线段树优化建图)
BZOJ UOJ 首先不考虑奇怪方格的限制,就是类似最大权闭合子图一样建图. 对于奇怪方格的影响,显然可以建一条边\((i\to x,p_i)\),然后由\(x\)向\(1\sim i-1\)中权值在 ...
- BZOJ - 2588 Spoj 10628. Count on a tree (可持久化线段树+LCA/树链剖分)
题目链接 第一种方法,dfs序上建可持久化线段树,然后询问的时候把两点之间的所有树链扒出来做差. #include<bits/stdc++.h> using namespace std; ...
- BZOJ 3439 Kpm的MCpassword Trie树+可持久化线段树
题目大意:给定n个字符串,对于每一个字符串求以这个字符串为后缀的字符串中第k小的编号 首先将字符串反转 那么就变成了对于每一个字符串求以这个字符串为前缀的字符串中第k小的编号 然后考虑对字符串排序 那 ...
- BZOJ 4771: 七彩树 可持久化线段树+树链的并
这个思路挺有意思的 ~ 利用树链的并来保证每个颜色只贡献一次,然后用可持久化线段树维护 code: #include <set> #include <cstdio> #incl ...
- 归并树 划分树 可持久化线段树(主席树) 入门题 hdu 2665
如果题目给出1e5的数据范围,,以前只会用n*log(n)的方法去想 今天学了一下两三种n*n*log(n)的数据结构 他们就是大名鼎鼎的 归并树 划分树 主席树,,,, 首先来说两个问题,,区间第k ...
随机推荐
- Chrome 清除某个特定网站下的缓存
打开开发者工具(F12),选择 Network--Disable cache 即可.需要清除某网站缓存时 F12 打开开发者工具就会自动清除这个网站的缓存,而不必清除所有网站的缓存了.
- 【bzoj3065】: 带插入区间K小值 详解——替罪羊套函数式线段树
不得不说,做过最爽的树套树———— 由于有了区间操作,我们很容易把区间看成一棵平衡树,对他进行插入,那么外面一层就是平衡树了,这就与我们之前所见到的不同了.我们之前所见到的大多数是线段树套平衡树而此题 ...
- spring学习之三 数据库操作jdbcTemplate
概念 jdbcTemplate就Spring对数据库持久化技术的实现,通过它可以对数据库进行CRUD等操作. JDBCTemplate和代码实现 public void jdbcadd() { Dri ...
- MySQL权限操作:Grant、Revoke
数据库操作: 创建数据库.创建表——CREATE 删除数据库.删除表——DROP 删除表内容——TRUNCATE.DELETE(后者效率低.一行一行地删除记录) 查询数据库.查询表——SELECT 插 ...
- WebApi Owin SelfHost OAuth2 - 授权服务和资源服务分离方案
使用JWT 参考:http://www.cnblogs.com/grissom007/p/6294746.html
- JsRender 前端渲染模板常用API学习
JsRender 常用API 1. $.templates() $.templates()方法是用来注册或编译模板的,使用的情况有以下几种. 把html字符串编译编译成模板 获取使用script标签声 ...
- T-sql语句修改数据库逻辑名、数据库名、物理名(sql2000)
--更改MSSQL数据库物理文件名Sql语句的写法 --注意:要在活动监视器里面确保没有进程连接你要改名的数据库!!!!!!!!!!!!!!!!!!!! -- Sql语句如下 USE master - ...
- LeetCode664. Strange Printer
There is a strange printer with the following two special requirements: The printer can only print a ...
- Linux命令之远程登录与执行远程主机命令
实现远程登录的命令 ssh.telnet.rlogin (1)ssh命令 ssh命令是openssh套件中的客户端连接工具,可以给予ssh加密协议实现安全的远程登录服务器.ssh命令用于远程登录上Li ...
- hdu 5112 (2014北京现场赛 A题)
给出某个时刻对应的速度 求出相邻时刻的平均速度 输出最大值 Sample Input23 // n2 2 //t v1 13 430 31 52 0 Sample OutputCase #1: 2.0 ...