bzoj4453
单调栈+set+后缀数组
一道奇妙的题
这道题如果对于每个询问$r$是固定的,那么就很简单了,可惜并不是
由于r会变化,那么对于两个子串$[i...r],[j...r]$,他们的大小关系随着r的变化也会变化,使得我们不能直接预处理答案
所以我们把询问离线,把每个询问按照r分类,通过考虑r的变化来完成询问
对于两个后缀$[i...r],[j...r],(i<j)$,设他们的$lcp=l$,那么当$l∈[j,j+l-1]$时,$[i...r]>[j...r]$,当$r=j+l$时,则变成$[i...r]<[j...r]$,因为$s[i+l]<s[j+l]$(假设)
我们称这样的两个后缀为$i$伴随$j$。
那么我们对答案维护一个set,里面存着一些后缀,保证后缀当前的大小单调递减,这样我们就可以每次快速询问答案。
但是由于$r$的变化每次大小关系会改变,也就是我们要把一些后缀删掉
那么我们再维护一个单调栈,这里存的是严格的后缀大小,这个单调栈是为了给每个新加入的后缀求出哪些后缀伴随他
那么如果当前的栈顶是大于新的后缀,$break$,否则标记$top$伴随$i$,并且记录当$i$,也就是$r=i+lcp$时删除栈顶
每次删除一个元素时,伴随他的元素也应该删除,所以每次$dfs$删除即可,并在$set$里删除
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <set>
#include <vector>
using namespace std;
const int N = 1e5 + ;
int n, k, m, top;
int rnk[N], tmp[N], sa[N], ans[N], vis[N], st[N], h[N][], Log[N];
char s[N];
set<int> S;
vector<pair<int, int> > v[N];
vector<int> del[N], g[N];
bool cmp(int i, int j) {
if(rnk[i] != rnk[j]) {
return rnk[i] < rnk[j];
}
int ri = i + k <= n ? rnk[i + k] : -;
int rj = j + k <= n ? rnk[j + k] : -;
return ri < rj;
}
void dfs(int u) {
vis[u] = ;
S.erase(u);
for(int i = ; i < g[u].size(); ++i) {
if(!vis[g[u][i]]) {
dfs(g[u][i]);
}
}
}
void SA() {
for(int i = ; i <= n; ++i) {
sa[i] = i;
rnk[i] = s[i];
}
for(k = ; k <= n; k <<= ) {
sort(sa + , sa + n + , cmp);
tmp[sa[]] = ;
for(int i = ; i <= n; ++i) {
tmp[sa[i]] = tmp[sa[i - ]] + (cmp(sa[i - ], sa[i]));
}
for(int i = ; i <= n; ++i) {
rnk[i] = tmp[i];
}
}
for(int i = ; i <= n; ++i) {
rnk[sa[i]] = i;
}
int d = ;
for(int i = ; i <= n; ++i) {
if(rnk[i] <= ) {
continue;
}
int j = sa[rnk[i] - ];
if(d) {
--d;
}
for(; i + d <= n && j + d <= n; ++d) {
if(s[i + d] != s[j + d]) {
break;
}
}
h[rnk[i] - ][] = d;
}
for(int j = ; j <= ; ++j) {
for(int i = ; i + ( << j) - <= n; ++i) {
h[i][j] = min(h[i][j - ], h[i + ( << (j - ))][j - ]);
}
}
}
int rmq(int l, int r) {
if(l == r) {
return n - l + ;
}
l = rnk[l];
r = rnk[r];
if(l > r) {
swap(l, r);
}
--r;
int x = Log[r - l + ];
return min(h[l][x], h[r - ( << x) + ][x]);
}
int main() {
scanf("%s%d", s + , &m);
n = strlen(s + );
for(int i = ; i <= n; ++i) {
Log[i] = Log[i >> ] + ;
}
SA();
for(int i = ; i <= m; ++i) {
int l, r;
scanf("%d%d", &l, &r);
v[r].push_back(make_pair(l, i));
}
for(int i = ; i <= n; ++i) {
S.insert(i);
while(top) {
int lcp = rmq(st[top], i);
if(s[st[top] + lcp] > s[i + lcp]) {
break;
}
del[i + lcp].push_back(st[top]);
g[i].push_back(st[top]);
--top;
}
st[++top] = i;
for(int j = ; j < del[i].size(); ++j) {
if(!vis[del[i][j]]) {
dfs(del[i][j]);
}
}
for(int j = ; j < v[i].size(); ++j) {
pair<int, int> x = v[i][j];
ans[x.second] = *S.lower_bound(x.first);
}
}
for(int i = ; i <= m; ++i) {
printf("%d\n", ans[i]);
}
return ;
}
bzoj4453的更多相关文章
- 【BZOJ4453】cys就是要拿英魂! 后缀数组+单调栈+set
[BZOJ4453]cys就是要拿英魂! Description pps又开始dota视频直播了!一群每天被pps虐的蒟蒻决定学习pps的操作技术,他们把pps在这局放的技能记录了下来,每个技能用一个 ...
随机推荐
- 深入Asyncio(十二)Asyncio与单元测试
Testing with asyncio 之前有说过应用开发者不需要将loop当作参数在函数间传递,只需要调用asyncio.get_event_loop()即可获得.但是在写单元测试时,可能会需要用 ...
- 深入浅出Stream和parallelStream
https://blog.csdn.net/darrensty/article/details/79283146
- 摩根大通银行被黑客攻克, ATM机/网银危在旦夕,winxp退市灾难来临了
winxp4月退市到如今还不到半年,就出现故障了 7600多万个消费者银行账户被黑.此外还有700万个小企业账户的信息也被黑客窃取,这个算不算灾难呢?假设等到银行业彻底崩溃,资金彻底丧失,那不仅仅是灾 ...
- go module
前言 go 1.5 引进了vendor管理工程依赖包,但是vendor的存放路径是在GOPATH底下,另外每个依赖还可以有自己的vendor,通常会弄得很乱,尽管dep管理工具可以将vendor平级化 ...
- python classmethod方法 和 staticmethod
classmethod() 是一个类方法,用来装饰对应的函数.被classmethod 装饰之后就无需实例化,也不需要在函数中传self,但是被装饰的函数第一个参数需要是cls来表示自身类.可以用来调 ...
- delphi Base64编码/解码及数据压缩/解压知识
一.Base64编码/解码 一般用到的是Delphi自带的单元EncdDecd,当然还有第三方提供的单元或控件,其中我所接触到的认为比较好的有Indy的TIdMimeEncode / TIdMimeD ...
- 【BZOJ2457】[BeiJing2011]双端队列 贪心+模拟
[BZOJ2457][BeiJing2011]双端队列 Description Sherry现在碰到了一个棘手的问题,有N个整数需要排序. Sherry手头能用的工具就是若 ...
- asp.net mvc 中"未找到路径“/favicon.ico”的控制器或该控制器未实现 IController。"
FavIcon.ico是一个特殊的文件,它是浏览器请求一个网站时出现的.某些浏览器在书签和收藏夹中使用这个图标.在与这些图标相关的网站被打开时,某些浏览器也在标题栏或浏览器标签中中显示这个图标. 当一 ...
- 九度OJ 1040:Prime Number(质数) (递归)
时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:5278 解决:2180 题目描述: Output the k-th prime number. 输入: k≤10000 输出: The k- ...
- CNN延拓至 复数域