解题思路:

fail树上用权值线段树合并求right/endpos集合,再用倍增找到待查询串对应节点,然后权值线段树求第k大。

#include<bits/stdc++.h>
using namespace std; typedef long long ll;
const int maxn=1e5+5;
int n,q;
char s[maxn]; namespace SegTree{
int sum[maxn*100],L[maxn*100],R[maxn*100];
int tot1;
int update(int rt,int l,int r,int pos,int val){
int nrt=++tot1;
L[nrt]=L[rt]; R[nrt]=R[rt]; sum[nrt]=sum[rt]+val;
if(l!=r){
int mid=(l+r)>>1;
if(pos<=mid)L[nrt]=update(L[rt],l,mid,pos,val);
else R[nrt]=update(R[rt],mid+1,r,pos,val);
}
return nrt;
}
int merge(int rt1,int rt2){
if(!rt1 || !rt2)return rt1|rt2;
int nrt=++tot1;
L[nrt]=L[rt1]; R[nrt]=R[rt1]; sum[nrt]=sum[rt1]+sum[rt2];
L[nrt]=merge(L[rt1],L[rt2]);
R[nrt]=merge(R[rt1],R[rt2]);
return nrt;
} int query(int rt,int l,int r,int k){
if(l==r)return l;
int mid=(l+r)>>1;
if(k<=sum[L[rt]])return query(L[rt],l,mid,k);
return query(R[rt],mid+1,r,k-sum[L[rt]]);
}
}using namespace SegTree; namespace Suffix_Automaton{
int ch[maxn<<1][26],fa[maxn<<1],len[maxn<<1];
int last,tot; int rt[maxn],T[maxn<<1];
int Fa[maxn<<1][20]; inline void init(){
last=tot=1;
len[1]=fa[0]=0;
memset(ch[1],0,sizeof(ch[1])); T[1]=0;
} inline int newnode(){
++tot;
len[tot]=fa[tot]=0;
memset(ch[tot],0,sizeof(ch[tot])); T[tot]=0;
return tot;
} inline void extend(int c,int right){
int p=last,cur=newnode();
len[cur]=len[last]+1;
last=cur; rt[right]=cur;
T[cur]=update(T[cur],1,n,right,1); while(p && !ch[p][c]){
ch[p][c]=cur;
p=fa[p];
}
if(!p)fa[cur]=1;
else{
int q=ch[p][c];
if(len[p]+1==len[q])fa[cur]=q;
else{
int clone=newnode();
len[clone]=len[p]+1;
memcpy(ch[clone],ch[q],sizeof(ch[q]));
fa[clone]=fa[q];
fa[q]=fa[cur]=clone;
while(ch[p][c]==q){
ch[p][c]=clone;
p=fa[p];
}
}
}
} int c[maxn<<1],A[maxn<<1];
inline void init(char *a,int l){
init();
for(int i=1;i<=l;i++)extend(a[i]-'a',i);
for(int i=0;i<=tot;i++)c[i]=0;
for(int i=1;i<=tot;i++)++c[len[i]];
for(int i=1;i<=tot;i++)c[i]+=c[i-1];
for(int i=1;i<=tot;i++)A[--c[len[i]]]=i;
for(int i=tot-1;i>=1;i--)T[fa[A[i]]]=merge(T[fa[A[i]]],T[A[i]]);
for(int i=1;i<=tot;i++)Fa[i][0]=fa[i];
for(int k=1;k<=19;k++)for(int i=1;i<=tot;i++)Fa[i][k]=Fa[Fa[i][k-1]][k-1]; } inline void solve(int l,int r,int k){
int u=rt[r],length=r-l+1;
if(len[fa[u]]+1>length){
for(int k=19;k>=0;k--)if(len[fa[Fa[u][k]]]+1>length)u=Fa[u][k];
u=fa[u];
}
if(k<=sum[T[u]])printf("%d\n",query(T[u],1,n,k)-length+1);
else printf("-1\n");
} }using namespace Suffix_Automaton;
int main()
{
//#ifndef ONLINE_JUDGE
// freopen("in.txt","r",stdin);
//#endif
int T;
scanf("%d",&T);
while(T--){
tot1=0;
scanf("%d %d",&n,&q);
scanf("%s",s+1);
init(s,n);
int l,r,k;
while(q--){
scanf("%d %d %d",&l,&r,&k);
solve(l,r,k);
}
}
return 0;
}

hdu6704 2019CCPC网络选拔赛1003 K-th occurrence 后缀自动机+线段树合并的更多相关文章

  1. hdu6704 2019CCPC网络选拔赛1003 K-th occurrence 后缀数组

    题意:给你一个长度为n的字符串,有q个询问,每次询问一个子串s(l,r)第k次出现的位置,若子串出现次数少于k次输出-1. 解题思路:先把SA跑出来,然后对于每次询问可以由l和rank[]找到l在所有 ...

  2. HDU - 6704 K-th occurrence (后缀数组+主席树/后缀自动机+线段树合并+倍增)

    题意:给你一个长度为n的字符串和m组询问,每组询问给出l,r,k,求s[l,r]的第k次出现的左端点. 解法一: 求出后缀数组,按照排名建主席树,对于每组询问二分或倍增找出主席树上所对应的的左右端点, ...

  3. HDU-6704 K-th occurrence (后缀自动机father树上倍增建权值线段树合并)

    layout: post title: HDU-6704 K-th occurrence (后缀自动机father树上倍增建权值线段树合并) author: "luowentaoaa&quo ...

  4. BZOJ 3065 带插入区间K小值(sag套线段树)

    3065: 带插入区间K小值 Time Limit: 60 Sec  Memory Limit: 512 MBSubmit: 4696  Solved: 1527[Submit][Status][Di ...

  5. [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)

    [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...

  6. SPOJ-COT-Count on a tree(树上路径第K小,可持久化线段树)

    题意: 求树上A,B两点路径上第K小的数 分析: 同样是可持久化线段树,只是这一次我们用它来维护树上的信息. 我们之前已经知道,可持久化线段树实际上是维护的一个前缀和,而前缀和不一定要出现在一个线性表 ...

  7. 树上第k小,可持久化线段树+倍增lca

    给定一颗树,树的每个结点都有权值, 有q个询问,每个询问是 u v k ,表示u到v路径上第k小的权值是多少. 每个结点所表示的线段树,是父亲结点的线段树添加该结点的权值之后形成的新的线段树 c[ro ...

  8. 网络赛 I题 Max answer 单调栈+线段树

    题目链接:https://nanti.jisuanke.com/t/38228 题意:在给出的序列里面找一个区间,使区间最小值乘以区间和得到的值最大,输出这个最大值. 思路:我们枚举每一个数字,假设是 ...

  9. ACM-ICPC 2018 南京赛区网络预赛 G Lpl and Energy-saving Lamps(线段树)

    题目链接:https://nanti.jisuanke.com/t/30996 中文题目: 在喝茶的过程中,公主,除其他外,问为什么这样一个善良可爱的龙在城堡里被监禁Lpl?龙神秘地笑了笑,回答说这是 ...

随机推荐

  1. ubuntu apt-get 安装找不到包问题

    1.首先 sudo gedit /etc/apt/sources.list  删除全部换成国内源 (推荐163) 2.考虑 ubuntu apt-get update失败 1.出现错误:E:Could ...

  2. 关于Exceptionless日志收集框架如何关闭磁盘缓存

    问题:在使用Exceptionless的时候,Exception在收集到日志时会默认在appdata里面缓存当条日志的缓存文件,时间久了之后,如果收集到的日志越多磁盘的空间就会不足. 我使用的环境是 ...

  3. .NET Core Web APi大文件分片上传研究

    前言 前两天发表利用FormData进行文件上传,然后有人问要是大文件几个G上传怎么搞,常见的不就是分片再搞下断点续传,动动手差不多也能搞出来,只不过要深入的话,考虑的东西还是很多.由于断点续传之前写 ...

  4. Integer.valueOf源码分析

    1. 引言 在牛客网上看到这样一道题目,判断一下打印的结果 public static void main(String[] args){ Integer i1 = 128; Integer i2 = ...

  5. 【模式识别与机器学习】——最大似然估计 (MLE) 最大后验概率(MAP)和最小二乘法

    1) 极/最大似然估计 MLE 给定一堆数据,假如我们知道它是从某一种分布中随机取出来的,可是我们并不知道这个分布具体的参,即“模型已定,参数未知”.例如,我们知道这个分布是正态分布,但是不知道均值和 ...

  6. JavaFX桌面应用-MVC模式开发,“真香”

    使用mvc模块开发JavaFX桌面应用在JavaFX系列文章第一篇 JavaFX桌面应用开发-HelloWorld 已经提到过,这里单独整理使用mvc模式开发开发的流程. ~ JavaFX桌面应用开发 ...

  7. 2020-06-20:一句话总结ZK?

    福哥答案2020-06-20: 这道题价值不大,但是面试题里有这道题. 分布式协调服务,注册服务和发现,树形结构,监听机制,过半机制. ZooKeeper是源代码开放的分布式协调服务,由雅虎公司创建, ...

  8. C#实践设计模式原则SOLID

    理论跟实践的关系,说远不远,说近不近.能不能把理论用到实践上,还真不好说.   通常讲到设计模式,一个最通用的原则是SOLID: S - Single Responsibility Principle ...

  9. 阿里天池 NLP 入门赛 TextCNN 方案代码详细注释和流程讲解

    thumbnail: https://image.zhangxiann.com/jung-ho-park-HbnqEhMBpPM-unsplash.jpg toc: true date: 2020/8 ...

  10. leetcode刷题记录——哈希表

    1.两数之和 可以先对数组进行排序,然后使用双指针方法或者二分查找方法.这样做的时间复杂度为 O(NlogN),空间复杂度为 O(1). 用 HashMap 存储数组元素和索引的映射,在访问到 num ...