Problem C: 重复子串(string)
/*
一个性质? right集合中只有相邻的位置才会有用
那么考虑set启发式合并, 能够求出大概nlogn个有用的对
那么将这些对按照右端点排序, 查询也按照右端点排序就可以离线维护信息
然后需要维护答案的话?? 区间和一次函数取max???
应该有别的办法吧
普通线段树就好了
*/
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<set>
#include<queue>
#include<iostream>
#define ll long long
#define mmp make_pair
#define M 200010
using namespace std;
int read() {
int nm = 0, f = 1;
char c = getchar();
for(; !isdigit(c); c = getchar()) if(c == '-') f = -1;
for(; isdigit(c); c = getchar()) nm = nm * 10 + c - '0';
return nm * f;
}
set<int> st[M];
set<int>::iterator it, tt;
int ch[M][26], fa[M], len[M], sz[M], ans[M], lst = 1, cnt = 1, n, m, tp;
char s[M];
vector<int> to[M];
struct Note {
int l, r, max;
bool operator < (const Note &b) const {
return this->r < b.r;
}
} note[M * 60], que[M];
void insert(int pl, int c) {
int p = ++cnt, f = lst;
lst = p;
len[p] = len[f] + 1;
while(f && !ch[f][c]) ch[f][c] = p, f = fa[f];
if(f == 0) fa[p] = 1;
else {
int q = ch[f][c];
if(len[q] == len[f] + 1) fa[p] = q;
else {
int nq = ++cnt;
len[nq] = len[f] + 1;
fa[nq] = fa[q];
memcpy(ch[nq], ch[q], sizeof(ch[q]));
fa[p] = fa[q] = nq;
while(f && ch[f][c] == q) ch[f][c] = nq, f = fa[f];
}
}
sz[p] = 1;
st[p].insert(pl);
}
int id[M];
bool cmp(int a, int b) {
return sz[a] > sz[b];
}
void dfs(int now) {
id[now] = now;
int flag = sz[now];
for(int i = 0; i < to[now].size(); i++) {
int vj = to[now][i];
dfs(vj);
sz[now] += sz[vj];
}
if(now == 1) return;
sort(to[now].begin(), to[now].end(), cmp);
if(to[now].size() != 0) {
id[now] = id[to[now][0]];
if(flag) {
int v = *st[now].begin();
it = st[id[now]].lower_bound(v);
if(it != st[id[now]].end()) {
note[++tp] = (Note) {
v, *it, len[now]
};
}
if(it != st[id[now]].begin()) {
it--;
note[++tp] = (Note) {
*it, v, len[now]
};
}
st[id[now]].insert(v);
}
for(int i = 1; i < to[now].size(); i++) {
int vj = to[now][i];
it = st[id[vj]].begin();
while(it != st[id[vj]].end()) {
int v = *it++;
tt = st[id[now]].lower_bound(v);
if(tt != st[id[now]].end()) {
note[++tp] = (Note) {
v, *tt, len[now]
};
}
if(tt != st[id[now]].begin()) {
tt--;
note[++tp] = (Note) {
*tt, v, len[now]
};
}
st[id[now]].insert(v);
}
}
}
}
struct Black_ {
#define ls now << 1
#define rs now << 1 | 1
#define lson l, mid, now << 1
#define rson mid + 1, r, now << 1 | 1
int t[M << 2], len[M << 2], ver[M << 2], maxx[M << 2];
void build(int l, int r, int now) {
len[now] = r - l + 1;
ver[now] = -0x3e3e3e3e;
if(l == r) return;
int mid = (l + r) >> 1;
build(lson), build(rson);
}
void add(int now, int v) {
ver[now] = max(ver[now], v);
maxx[now] = max(maxx[now], ver[now] + len[now]);
}
void pushup(int now) {
maxx[now] = max(maxx[now], max(maxx[ls], maxx[rs]));
}
void pushdown(int now) {
add(rs, ver[now]);
add(ls, ver[now] + len[rs]);
}
int que(int l, int r, int now, int ln, int rn) {
if(l > rn || r < ln) return 0;
if(l >= ln && r <= rn) return maxx[now];
int mid = (l + r) >> 1;
pushdown(now);
return max(que(lson, ln, rn), que(rson, ln, rn));
}
void modi(int l, int r, int now, int ln, int rn, int v) {
if(l > rn || r < ln) return;
if(l >= ln && r <= rn) {
add(now, v);
return;
}
int mid = (l + r) >> 1;
pushdown(now);
modi(lson, ln, rn, v + min(max(rn - mid, 0), len[rs]));
modi(rson, ln, rn, v);
pushup(now);
}
void modify(int r, int v) {
modi(1, n, 1, r - v + 1, r, 0);
}
int query(int l, int r) {
return que(1, n, 1, l, r);
}
} Seg;
int main() {
// freopen("string_example_3.in", "r", stdin); freopen("zz.out", "w", stdout);
n = read(), m = read();
scanf("%s", s + 1);
for(int i = 1; i <= n; i++) insert(i, s[i] - 'a');
for(int i = 2; i <= cnt; i++) to[fa[i]].push_back(i);
dfs(1);
for(int i = 1; i <= m; i++) que[i].l = read(), que[i].r = read(), que[i].max = i;
sort(note + 1, note + tp + 1);
// puts(s + 1);
// for(int i = 1; i <= tp; i++) {
// cerr<< note[i].l << " " << note[i].r << " " << note[i].max << "\n";
// }
// cerr << tp << "\n";
sort(que + 1, que + m + 1);
int tp1 = 1, tp2 = 1;
Seg.build(1, n, 1);
for(int i = 1; i <= n; i++) {
while(tp1 <= tp && note[tp1].r <= i) {
Seg.modify(note[tp1].l, note[tp1].max);
tp1++;
}
while(tp2 <= m && que[tp2].r <= i) {
ans[que[tp2].max] = Seg.query(que[tp2].l, que[tp2].r);
tp2++;
}
}
for(int i = 1; i <= m; i++) cout << ans[i] << "\n";
return 0;
}
Problem C: 重复子串(string)的更多相关文章
- POJ 1743 (后缀数组+不重叠最长重复子串)
题目链接: http://poj.org/problem?id=1743 题目大意:楼教主の男人八题orz.一篇钢琴谱,每个旋律的值都在1~88以内.琴谱的某段会变调,也就是说某段的数可以加减一个旋律 ...
- POJ 1743 Musical Theme (后缀数组,求最长不重叠重复子串)(转)
永恒的大牛,kuangbin,膜拜一下,Orz 链接:http://www.cnblogs.com/kuangbin/archive/2013/04/23/3039313.html Musical T ...
- poj 1743 男人八题之后缀数组求最长不可重叠最长重复子串
Musical Theme Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 14874 Accepted: 5118 De ...
- hdu_2668 Daydream O(n)求最长不重复子串
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2668 Daydream Time Limit: 2000/1000 MS (Java/Others) ...
- [Jobdu] 题目1530:最长不重复子串
题目描述: 最长不重复子串就是从一个字符串中找到一个连续子串,该子串中任何两个字符都不能相同,且该子串的长度是最大的. 输入: 输入包含多个测试用例,每组测试用例输入一行由小写英文字符a,b,c... ...
- ACdream 1430——SETI——————【后缀数组,不重叠重复子串个数】
SETI Time Limit: 4000/2000MS (Java/Others) Memory Limit: 128000/64000KB (Java/Others) Submit Statist ...
- POJ3693 Maximum repetition substring —— 后缀数组 重复次数最多的连续重复子串
题目链接:https://vjudge.net/problem/POJ-3693 Maximum repetition substring Time Limit: 1000MS Memory Li ...
- SPOJ - REPEATS —— 后缀数组 重复次数最多的连续重复子串
题目链接:https://vjudge.net/problem/SPOJ-REPEATS REPEATS - Repeats no tags A string s is called an (k,l ...
- [LeetCode] Longest Substring Without Repeating Characters 最长无重复子串
Given a string, find the length of the longest substring without repeating characters. For example, ...
随机推荐
- 跟随我在oracle学习php(9)
三目运算符:表达式? 表达式: 表达式: 自增在前在后没有影响 参与表达式需要注意 在前先计算,在后最后加1. + 字符串拼接. 字符串转数字:从左到右第一个不是数字的位置结束 取整 parseInt ...
- C++中模板的特化与偏特化
1.引言 C++中的模板分为类模板和函数模板,虽然它引进到C++标准中的时间不是很长,但是却得到了广泛的应用,这一点在STL中有着充分的体现.目前,STL在C++社区中得到了广泛的关注.应用和研究.理 ...
- js 面试知识点
基础 原型 原型链 作用域 闭包 异步 单线程 JS API DOM操作 AJAX 事件绑定 开发环境 版本管理 模块化 打包工具 运行环境 页面渲 ...
- intelij idea常用功能介绍
1.查看本地文件修改记录 保存本地修改记录: 可以将system下的LocalHistory保存,到另一个目录,需要的时候保存即可. 2.debbuger查看代码 1)优化设置 2)常用 3.条件断点 ...
- python基础提高演练(名片管理系统的开发)
综合应用 —— 名片管理系统 目标 综合应用已经学习过的知识点: 变量 流程控制 函数 模块 开发 名片管理系统 系统需求 1. 程序启动,显示名片管理系统欢迎界面,并显示功能菜单 ********* ...
- Van Emde Boas Tree
van Emde Boas trees 支持所有优先级优先级队列的操作,并且巧妙的是它对于SEARCH, INSERT,DELETE,MINIMUM,MAXMUN,SUCCESSOR,和PREDECE ...
- 剑指Offer 29. 最小的K个数 (其他)
题目描述 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4. 题目地址 https://www.nowcoder.com/prac ...
- CPU查询
请问:两路四核超线程=多少CPU??? 答案:16个 一.解释说明 两路=物理主机中,CPU的卡槽,槽位,这个无法增加,一个萝卜一个坑.就更内存卡槽一样,两个内存卡卡槽,a+b=总内存. 路=槽位=插 ...
- yii2 Rbac实例 (做完以下这些 会有些小的报错,相信各位都能解决,大多数都是自己命名空间上的问题)。
首先我自己没有使用自带的user表 如果model层没有AuthItem.php 那就自建一个将下面这些内容写入 <?php namespace backend\models; use Yi ...
- python set 集合复习--点滴
一.set特性: set是一个无序不重复的元素集合. 集合对象是一组无序排列的可哈希的值,集合成员可以做字典中的键.集合支持用in和not in操作符检查成员,由len()内建函数得到集合的基数(大小 ...