K-th occurrence HDU - 6704 (SA, 主席树)
大意: 给定串$s$, $q$个询问$(l,r,k)$, 求子串$s[l,r]$的第$k$次出现位置.
本来是个简单签到题, 可惜比赛的时候还没学$SA$...... 好亏啊
相同的子串在$SA$中是一定是连续的一段$[L,R]$
满足对于$L<i\le R$都有$h_i\ge r-l+1$
可以先用线段树二分出$L,R$, 然后主席树查询第$k$大即可
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <cstring>
#define REP(i,a,n) for(int i=a;i<=n;++i)
#define PER(i,a,n) for(int i=n;i>=a;--i)
#define lc (o<<1)
#define rc (lc|1)
#define mid ((l+r)>>1)
#define ls lc,l,mid
#define rs rc,mid+1,r
using namespace std;
const int N = 1e5+10;
int n, q, tot, a[N], T[N];
struct {int l,r,v;} tr[N*40];
char s[N];
int c[N],rk[N],h[N],sa[N],mi[N<<2]; void build(int *a, int n, int m) {
a[n+1] = 0;
int i,*x=rk,*y=h;
for(i=1;i<=m;i++) c[i]=0;
for(i=1;i<=n;i++) c[x[i]=a[i]]++;
for(i=1;i<=m;i++) c[i]+=c[i-1];
for(i=n;i;i--) sa[c[x[i]]--]=i;
for(int k=1,p;k<=n;k<<=1) {
p=0;
for(i=n-k+1;i<=n;i++) y[++p]=i;
for(i=1;i<=n;i++) if(sa[i]>k) y[++p]=sa[i]-k;
for(i=1;i<=m;i++) c[i]=0;
for(i=1;i<=n;i++) c[x[y[i]]]++;
for(i=1;i<=m;i++) c[i]+=c[i-1];
for(i=n;i;i--) sa[c[x[y[i]]]--]=y[i];
swap(x,y); x[sa[1]]=1; p=1;
for(i=2;i<=n;i++)
x[sa[i]]=(y[sa[i-1]]==y[sa[i]] && y[sa[i-1]+k]==y[sa[i]+k])?p:++p;
if(p==n) break; m=p;
}
for(i=1;i<=n;i++) rk[sa[i]]=i;
for(int i=1,j,k=0;i<=n;i++){
if(k) k--;
j=sa[rk[i]-1];
while (a[i+k]==a[j+k]) k++;
h[rk[i]] = k;
}
}
//求最小的位置p, 使得[p,x]的最小值>=v
int find1(int o, int l, int r, int x, int v) {
if (r<=x) {
if (l==r) return mi[o]>=v?l:-1;
if (mi[rc]<v) return find1(rs,x,v);
int t = find1(ls,x,v);
return t==-1?mid+1:t;
}
if (mid>=x) return find1(ls,x,v);
int R = find1(rs,x,v);
if (R==-1||R>mid+1) return R;
int L = find1(ls,x,v);
return L==-1?R:L;
}
//求找最大的位置p, 使得[x,p]的最小值>=v
int find2(int o, int l, int r, int x, int v) {
if (x<=l) {
if (l==r) return mi[o]>=v?l:-1;
if (mi[lc]<v) return find2(ls,x,v);
int t = find2(rs,x,v);
return t==-1?mid:t;
}
if (mid<x) return find2(rs,x,v);
int L = find2(ls,x,v);
if (L==-1||L<mid) return L;
int R = find2(rs,x,v);
return R==-1?L:R;
}
int query(int u, int v, int l, int r, int k) {
if (l==r) return l;
int s = tr[tr[v].l].v-tr[tr[u].l].v;
if (s>=k) return query(tr[u].l,tr[v].l,l,mid,k);
return query(tr[u].r,tr[v].r,mid+1,r,k-s);
}
void add(int &o, int l, int r, int x) {
tr[++tot]=tr[o],o=tot,++tr[o].v;
if (l!=r) mid>=x?add(tr[o].l,l,mid,x):add(tr[o].r,mid+1,r,x);
}
int query(int p, int len, int k) {
int l = p>1?find1(1,2,n,p,len)-1:1;
int r = p<n?find2(1,2,n,p+1,len):n;
if (l<0) l = p;
if (r<0) r = p;
if (r-l+1>=k) return query(T[l-1],T[r],1,n,k);
return -1;
}
void build2(int o, int l, int r) {
if (l==r) return mi[o]=h[l],void();
build2(ls),build2(rs);
mi[o]=min(mi[lc],mi[rc]);
}
void brute_force() {
while (q--) {
int l, r, k;
scanf("%d%d%d",&l,&r,&k);
string g(s+l,s+r+1);
int pos = -1, cnt = 0;
REP(i,1,n) if (string(s+i,s+i+r-l+1)==g) {
if (++cnt==k) {
pos = i; break;
}
}
printf("%d\n", pos);
}
}
int main() {
int t;
scanf("%d", &t);
while (t--) {
scanf("%d%d%s", &n, &q, s+1);
if (n<=10) {brute_force();continue;}
REP(i,1,n) a[i]=s[i]-'a'+1;
build(a,n,26);
build2(1,2,n);
REP(i,1,n) {
T[i] = T[i-1];
add(T[i],1,n,sa[i]);
}
while (q--) {
int l, r, k;
scanf("%d%d%d", &l, &r, &k);
int len = r-l+1;
printf("%d\n", query(rk[l],r-l+1,k));
}
REP(i,0,n) T[i]=0;
while (tot) tr[tot].l=tr[tot].r=tr[tot].v=0,--tot;
}
}
K-th occurrence HDU - 6704 (SA, 主席树)的更多相关文章
- 2019CCPC网络赛 C - K-th occurrence HDU - 6704(后缀数组+ST表+二分+主席树)
题意 求区间l,r的子串在原串中第k次出现的位置. 链接:https://vjudge.net/contest/322094#problem/C 思路 比赛的时候用后缀自动机写的,TLE到比赛结束. ...
- HDU-6704 K-th occurrence(后缀数组+主席树)
题意 给一个长度为n的字符串,Q次询问,每次询问\((l,r,k)\) , 回答子串\(s_ls_{l+1}\cdots s_r\) 第\(k\) 次出现的位置,若不存在输出-1.\(n\le 1e5 ...
- 洛谷 P4094 [HEOI2016/TJOI2016]字符串(SA+主席树)
题面传送门 一道码农题---- u1s1 感觉这类题目都挺套路的,就挑个有代表性的题写一篇题解罢. 首先注意到答案满足可二分性,故考虑二分答案 \(mid\),转化为判定性问题. 考虑怎样检验 \(m ...
- Sequence II HDU - 5919(主席树)
Mr. Frog has an integer sequence of length n, which can be denoted as a1,a2,⋯,ana1,a2,⋯,anThere are ...
- Super Mario HDU - 4417 (主席树)
Mario is world-famous plumber. His “burly” figure and amazing jumping ability reminded in our memory ...
- 【10.10校内测试】【线段树维护第k小+删除】【lca+主席树维护前驱后驱】
贪心思想.将a排序后,对于每一个a,找到对应的删除m个后最小的b,每次更新答案即可. 如何删除才是合法并且最优的?首先,对于排了序的a,第$i$个那么之前就应该删除前$i-1$个a对应的b.剩下$m- ...
- 静态区间第K小(整体二分、主席树)
题目链接 题解 主席树入门题 但是这里给出整体二分解法 整体二分顾名思义是把所有操作放在一起二分 想想,如果求\([1-n]\)的第\(k\)小怎么二分求得? 我们可以二分答案\(k\), \(O(n ...
- HDU 3727 Jewel 主席树
题意: 一开始有一个空序列,然后有下面四种操作: Insert x在序列尾部加入一个值为\(x\)的元素,而且保证序列中每个元素都互不相同. Query_1 s t k查询区间\([s,t]\)中第\ ...
- 2019CCPC网络预选赛 1003 K-th occurrence 后缀自动机 + 二分 + 主席树
题意:给你一个长度为n的字符串,有m次询问,每次询问l到r的子串在原串中第k次出现的位置,如果没有输出-1.n, m均为1e5级别. 思路:后悔没学后缀数组QAQ,其实只要学过后缀数组这个题还是比较好 ...
随机推荐
- 修改layui的表单手机、邮箱验证可以为空怎么实现?
修改layui的表单手机.邮箱验证可以为空 解决办法: 修改源码: 把表单验证源代码(form.js)的正则表达式改一下,例如手机的正则为:/^1d{10}$/,可以改成/^$|^1d{10} ...
- php foreach 无法改变数组的值的问题
转:http://www.cnblogs.com/yangwenxin/p/5845212.html 翻到PHP文档的foreach那页这样写道: “foreach 语法结构提供了遍历数组的简单方式. ...
- Hive的配置详解和日常维护
Hive的配置详解和日常维护 一.Hive的参数配置详解 1>.mapred.reduce.tasks 默认为-1.指定Hive作业的reduce task个数,如果保留默认值,则Hive 自 ...
- js监听浏览器剪贴板
function setClipboardText(event){ event.preventDefault(); var node = document.createElement('div'); ...
- c代码审查软件
1. Coccinelle http://coccinelle.lip6.fr/
- RabbitMQ 入门教程(PHP版) 第一部分:Hello World
abbitMQ是一个消息代理.它的核心原理非常简单:接收和发送消息.你可以把它想像成一个邮局:你把信件放入邮箱,邮递员就会把信件投递到你的收件人处.在这个比喻中,RabbitMQ是一个邮箱.邮局.邮递 ...
- LeetCode_283. Move Zeroes
283. Move Zeroes Easy Given an array nums, write a function to move all 0's to the end of it while m ...
- DataTable.NET 使用server-side processing
https://datatables.net/examples/server_side/simple.html 當頁面上要顯示的數據在10萬筆以上時,可以使用server-side processin ...
- vuex中mutations数据响应
vuex中的mutation需遵守Vue的响应规则: 既然Vuex的store中的状态是响应式的,那么在我们变更状态时,监视状态的Vuex最好在state中初始化好所有的所需属性. 如果需要在对象上添 ...
- skynet sproto 问题
刚碰到一个小细节,纠结了半个小时 sproto的协议,request 和{ 必须有空格