正题

题目链接: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,线段树合并】的更多相关文章

  1. [NOI2018]你的名字(SAM+线段树合并)

    考虑l=1,r=n的68分,对S和T建SAM,对T的SAM上的每个节点,计算它能给答案带来多少贡献. T上节点x代表的本质不同的子串数为mx[x]-mx[fa[x]],然后需要去掉所代表子串与S的最长 ...

  2. NOI2018 你的名字——SAM+线段树合并

    题目链接在这里洛谷/LOJ 题目大意 有一个串\(S\),每次询问给你一个串\(T\),两个数\(L\)和\(R\),问你\(T\)有多少个本质不同的子串不是\(S[L,R]\)的子串 SOLUTIO ...

  3. 【BZOJ5417】[NOI2018]你的名字(线段树,后缀自动机)

    [BZOJ5417][NOI2018]你的名字(线段树,后缀自动机) 题面 BZOJ 洛谷 题解 首先考虑\(l=1,r=|S|\)的做法,对于每次询问的\(T\)串,暴力在\(S\)串的\(SAM\ ...

  4. 【NOI2018】你的名字(SAM & 线段树合并)

    Description Hint Solution 不妨先讨论一下无区间限制的做法. 首先"子串"可以理解为"前缀的后缀",因此我们定义一个 \(\lim(i) ...

  5. UOJ#395. 【NOI2018】你的名字 字符串,SAM,线段树合并

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ395.html 题解 记得同步赛的时候这题我爆0了,最暴力的暴力都没调出来. 首先我们看看 68 分怎么做 ...

  6. luogu4770 [NOI2018]你的名字 (SAM+主席树)

    对S建SAM,拿着T在上面跑 跑的时候不仅无法转移要跳parent,转移过去不在范围内也要跳parent(注意因为范围和长度有关,跳的时候应该把长度一点一点地缩) 这样就能得到对于T的每个前缀,它最长 ...

  7. CF1037H Security——SAM+线段树合并

    又是一道\(SAM\)维护\(endpos\)集合的题,我直接把CF700E的板子粘过来了QwQ 思路 如果我们有\([l,r]\)对应的\(SAM\),只需要在上面贪心就可以了.因为要求的是字典序比 ...

  8. 洛谷P4482 [BJWC2018]Border 的四种求法 字符串,SAM,线段树合并,线段树,树链剖分,DSU on Tree

    原文链接https://www.cnblogs.com/zhouzhendong/p/LuoguP4482.html 题意 给定一个字符串 S,有 q 次询问,每次给定两个数 L,R ,求 S[L.. ...

  9. Codeforces 700E. Cool Slogans 字符串,SAM,线段树合并,动态规划

    原文链接https://www.cnblogs.com/zhouzhendong/p/CF700E.html 题解 首先建个SAM. 一个结论:对于parent树上任意一个点x,以及它所代表的子树内任 ...

随机推荐

  1. c# 对 struct为什么不能继承类和结构的思考

    1.类.结构在使用的时候可以不调用构造函数,如果能够继承类,这种情况下不能够初始化基类,因为不执行构造函数 2.结构.所有的结构类型都默认是 sealed,通过 反汇编可以看到  ,这就阻止了结构被继 ...

  2. Spring整合Quartz分布式定时任务

    概述虽然单个Quartz实例能给予你很好的Job调度能力,但它不能满足典型的企业需求,如可伸缩性.高可靠性满足.假如你需要故障转移的能力并能运行日益增多的 Job,Quartz集群势必成为你应用的一部 ...

  3. 【nodejs】express框架+mysql后台数据查询

    一 环境部署 1,首先安装nodejs,并配置好环境变量(看自己习惯), 2,安装Express npm install express -g //全局安装 npm install express-g ...

  4. [SWMM]模型子汇水区划分的几种方法

    子汇水区的划分是SWMM模型建模的主要步骤之一,划分的好坏对结果精度有比较大的影响.概括来讲,子汇水区的划分有以下几种思路: (1)根据管网走向.建筑物和街道分布,直接人工划分子汇水区.这个方法适用于 ...

  5. 如何打一个RPM包

    如何打一个RPM包 参考链接:RPM打包原理.示例.详解及备查 前言 本文只是一个RPM安装的例子,并没有对RPM做比较详尽的叙述,更为详尽的讲解,可以在上面的链接中找到. RPM是啥? RPM(Re ...

  6. 【Google Cloud技术咨询】「Contact Center AI」引领我们走向高度智能客服的时代

    前提背景 我们距离"不再智障"的智能客服还有多远?对于智能客服,用户一直都是"批评多于褒奖",究其原因是在于人们对于AI客服的期待很高,而AI客服在实际应用中的 ...

  7. go协程调度

    目录 前言 1. 线程池的缺陷 2.Goroutine 调度器 3.调度策略 3.1 队列轮转 3.2 系统调用 3.3 工作量窃取 4.GOMAXPROCS设置对性能的影响 参考 前言 Gorout ...

  8. 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 ...

  9. Windows内核-7-IRP和派遣函数

    Windows内核-7-IRP和派遣函数 IRP以及派遣函数是Windows中非常重要的概念.IRP 是I/O Request Pocket的简称,意思是I/O操作的请求包,Windows中所有Use ...

  10. 剑指 Offer 31. 栈的压入、弹出序列

    剑指 Offer 31. 栈的压入.弹出序列 输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否为该栈的弹出顺序.假设压入栈的所有数字均不相等.例如,序列 {1,2,3,4,5} 是某 ...