P4770-[NOI2018]你的名字【SAM,线段树合并】
正题
题目链接:https://www.luogu.com.cn/problem/P4770
题目大意
给出一个长度为\(n\)的字符串\(S\)。\(q\)次询问给出一个串\(T\)和一个区间\([L,R]\),求\(T\)有多少个本质不同的子串不是\(S_{L\sim R}\)的子串。
\(1\leq n\leq 5\times 10^5,1\leq Q\le 10^5,\sum|T|\leq 10^6\)
解题思路
因为给了很多\(L=1,R=n\)的部分分所以应该是提示我们先从这个方面考虑。
这个部分比较简单,考虑改为求有多少个本质不同的子串在\(S_{L\sim R}\)中出现过,因为是本质不同的子串,我们可以先建一个\(S\)的\(SAM\)和一个\(T\)的\(SAM\)。
然后把\(T\)串拿到\(S\)的\(SAM\)上面跑,然后每次跑出来的一个匹配长度记为\(len\)。对于\(T\)的\(SAM\)上的每一个节点我们记录一个\(pos\)表示这个节点属于的长度位置,然后跑到这个位置的\(len\)就是能够匹配的长度了,记为\(ans\)。然后答案就是\(max\{len_i-max\{ans_{pos_i},len_{fa_i}\},0\}\)(防匹配长度超出\([len_{fa_i},len_i]\)的范围)
这样一次的时间复杂度就是\(O(|T|)\)的了。
然后考虑带区间的怎么做,我们需要保证我们在\(S\)的\(SAM\)上面跳的时候需要保证这些节点都是属于\(S_{L\sim R}\)的自动机上的,而且要保证我们提取出来的\(len\)也是在那个上面的。
其实如果这个节点的\(endpos\)类里面有\(L\sim R\)的信息就好了,这个用线段树合并维护一下\(endpos\)类的信息就可以了。
时间复杂度\(O((n+\sum |T|)\log n)\)
code
#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
const int N=1e6+10;
int n,q,ql,qr,p[N],c[N],rt[N],ans[N],pos[N];
char s[N];
struct SegTree{
int cnt,w[N<<5],ls[N<<5],rs[N<<5];
int Change(int x,int L,int R,int pos){
int p=++cnt;w[p]=max(w[x],pos);
if(L==R)return p;int mid=(L+R)>>1;
if(pos<=mid)ls[p]=Change(ls[x],L,mid,pos),rs[p]=rs[x];
else rs[p]=Change(rs[x],mid+1,R,pos),ls[p]=ls[x];
return p;
}
int Ask(int x,int L,int R,int l,int r){
if(!x)return 0;
if(L==l&&R==r)return w[x];
int mid=(L+R)>>1;
if(r<=mid)return Ask(ls[x],L,mid,l,r);
if(l>mid)return Ask(rs[x],mid+1,R,l,r);
return max(Ask(ls[x],L,mid,l,mid),Ask(rs[x],mid+1,R,mid+1,r));
}
int Merge(int x,int y){
if(!x||!y)return x+y;
int p=++cnt;w[p]=max(w[x],w[y]);
ls[p]=Merge(ls[x],ls[y]);
rs[p]=Merge(rs[x],rs[y]);
return p;
}
}R;
struct SAM{
int cnt,last,ch[N][26],len[N],fa[N];
void init(){
memset(ch[1],0,sizeof(ch[1]));
last=cnt=1;
}
int Insert(int c){
int p=last,np=last=++cnt;
len[np]=len[p]+1;
memset(ch[np],0,sizeof(ch[np]));
for(;p&&!ch[p][c];p=fa[p])ch[p][c]=np;
if(!p)fa[np]=1;
else{
int q=ch[p][c];
if(len[p]+1==len[q])fa[np]=q;
else{
int nq=++cnt;len[nq]=len[p]+1;
memcpy(ch[nq],ch[q],sizeof(ch[nq]));
fa[nq]=fa[q];fa[q]=fa[np]=nq;pos[nq]=pos[q];
for(;p&&ch[p][c]==q;p=fa[p])ch[p][c]=nq;
}
}
last=np;return np;
}
void Build(){
for(int i=1;i<=cnt;i++)c[len[i]]++;
for(int i=1;i<=n;i++)c[i]+=c[i-1];
for(int i=1;i<=cnt;i++)p[c[len[i]]--]=i;
for(int i=cnt;i>=1;i--){
int x=p[i];
rt[fa[x]]=R.Merge(rt[fa[x]],rt[x]);
}
return;
}
void next(int &x,int &l,int c){
while(x){
if(ch[x][c]){
int maxl=R.Ask(rt[ch[x][c]],1,n,1,qr)-ql+1;
if(len[fa[x]]<maxl){l=min(l+1,maxl);x=ch[x][c];return;}
}
x=fa[x];l=len[x];
}
l=0;x=1;return;
}
ll calc(){
ll prt=0;
for(int i=2;i<=cnt;i++)
prt+=max(len[i]-max(ans[pos[i]],len[fa[i]]),0);
return prt;
}
}S,T;
void work(char *s){
int m=strlen(s+1);T.init();
for(int i=1;i<=m;i++){
int x=T.Insert(s[i]-'a');
pos[x]=i;
}
int x=1,l=0;
for(int i=1;i<=m;i++){
int c=s[i]-'a';
S.next(x,l,c);
ans[i]=l;
}
printf("%lld\n",T.calc());
}
signed main()
{
scanf("%s",s+1);n=strlen(s+1);
S.init();
for(int i=1;i<=n;i++){
int x=S.Insert(s[i]-'a');
rt[x]=R.Change(rt[x],1,n,i);
}
S.Build();
scanf("%d",&q);
while(q--){
scanf("%s",s+1);
scanf("%d%d",&ql,&qr);
work(s);
}
return 0;
}
P4770-[NOI2018]你的名字【SAM,线段树合并】的更多相关文章
- [NOI2018]你的名字(SAM+线段树合并)
考虑l=1,r=n的68分,对S和T建SAM,对T的SAM上的每个节点,计算它能给答案带来多少贡献. T上节点x代表的本质不同的子串数为mx[x]-mx[fa[x]],然后需要去掉所代表子串与S的最长 ...
- NOI2018 你的名字——SAM+线段树合并
题目链接在这里洛谷/LOJ 题目大意 有一个串\(S\),每次询问给你一个串\(T\),两个数\(L\)和\(R\),问你\(T\)有多少个本质不同的子串不是\(S[L,R]\)的子串 SOLUTIO ...
- 【BZOJ5417】[NOI2018]你的名字(线段树,后缀自动机)
[BZOJ5417][NOI2018]你的名字(线段树,后缀自动机) 题面 BZOJ 洛谷 题解 首先考虑\(l=1,r=|S|\)的做法,对于每次询问的\(T\)串,暴力在\(S\)串的\(SAM\ ...
- 【NOI2018】你的名字(SAM & 线段树合并)
Description Hint Solution 不妨先讨论一下无区间限制的做法. 首先"子串"可以理解为"前缀的后缀",因此我们定义一个 \(\lim(i) ...
- UOJ#395. 【NOI2018】你的名字 字符串,SAM,线段树合并
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ395.html 题解 记得同步赛的时候这题我爆0了,最暴力的暴力都没调出来. 首先我们看看 68 分怎么做 ...
- luogu4770 [NOI2018]你的名字 (SAM+主席树)
对S建SAM,拿着T在上面跑 跑的时候不仅无法转移要跳parent,转移过去不在范围内也要跳parent(注意因为范围和长度有关,跳的时候应该把长度一点一点地缩) 这样就能得到对于T的每个前缀,它最长 ...
- CF1037H Security——SAM+线段树合并
又是一道\(SAM\)维护\(endpos\)集合的题,我直接把CF700E的板子粘过来了QwQ 思路 如果我们有\([l,r]\)对应的\(SAM\),只需要在上面贪心就可以了.因为要求的是字典序比 ...
- 洛谷P4482 [BJWC2018]Border 的四种求法 字符串,SAM,线段树合并,线段树,树链剖分,DSU on Tree
原文链接https://www.cnblogs.com/zhouzhendong/p/LuoguP4482.html 题意 给定一个字符串 S,有 q 次询问,每次给定两个数 L,R ,求 S[L.. ...
- Codeforces 700E. Cool Slogans 字符串,SAM,线段树合并,动态规划
原文链接https://www.cnblogs.com/zhouzhendong/p/CF700E.html 题解 首先建个SAM. 一个结论:对于parent树上任意一个点x,以及它所代表的子树内任 ...
随机推荐
- c# 对 struct为什么不能继承类和结构的思考
1.类.结构在使用的时候可以不调用构造函数,如果能够继承类,这种情况下不能够初始化基类,因为不执行构造函数 2.结构.所有的结构类型都默认是 sealed,通过 反汇编可以看到 ,这就阻止了结构被继 ...
- Spring整合Quartz分布式定时任务
概述虽然单个Quartz实例能给予你很好的Job调度能力,但它不能满足典型的企业需求,如可伸缩性.高可靠性满足.假如你需要故障转移的能力并能运行日益增多的 Job,Quartz集群势必成为你应用的一部 ...
- 【nodejs】express框架+mysql后台数据查询
一 环境部署 1,首先安装nodejs,并配置好环境变量(看自己习惯), 2,安装Express npm install express -g //全局安装 npm install express-g ...
- [SWMM]模型子汇水区划分的几种方法
子汇水区的划分是SWMM模型建模的主要步骤之一,划分的好坏对结果精度有比较大的影响.概括来讲,子汇水区的划分有以下几种思路: (1)根据管网走向.建筑物和街道分布,直接人工划分子汇水区.这个方法适用于 ...
- 如何打一个RPM包
如何打一个RPM包 参考链接:RPM打包原理.示例.详解及备查 前言 本文只是一个RPM安装的例子,并没有对RPM做比较详尽的叙述,更为详尽的讲解,可以在上面的链接中找到. RPM是啥? RPM(Re ...
- 【Google Cloud技术咨询】「Contact Center AI」引领我们走向高度智能客服的时代
前提背景 我们距离"不再智障"的智能客服还有多远?对于智能客服,用户一直都是"批评多于褒奖",究其原因是在于人们对于AI客服的期待很高,而AI客服在实际应用中的 ...
- go协程调度
目录 前言 1. 线程池的缺陷 2.Goroutine 调度器 3.调度策略 3.1 队列轮转 3.2 系统调用 3.3 工作量窃取 4.GOMAXPROCS设置对性能的影响 参考 前言 Gorout ...
- unexpected end of file while looking for precompiled header. Did you forget to add '#include "stdafx.h"' to your source 解决办法
Project -> Properties -> C/C++ -> Precompiled Headers -> Precompiled Header -> 选择Not ...
- Windows内核-7-IRP和派遣函数
Windows内核-7-IRP和派遣函数 IRP以及派遣函数是Windows中非常重要的概念.IRP 是I/O Request Pocket的简称,意思是I/O操作的请求包,Windows中所有Use ...
- 剑指 Offer 31. 栈的压入、弹出序列
剑指 Offer 31. 栈的压入.弹出序列 输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否为该栈的弹出顺序.假设压入栈的所有数字均不相等.例如,序列 {1,2,3,4,5} 是某 ...