[bzoj3998][TJOI2015]弦论-后缀自动机
Brief Description
给定一个字符串, 您需要求出他的严格k小子串或非严格k小子串.
Algorithm Design
考察使用后缀自动机.
首先原串建SAM, 然后如果考察每个状态代表的子串的出现次数. 可以知道, 在parent树上某个节点的出现次数就是他的所有儿子的出现次数之和. 所以, 我们可以按照len基数排序, 然后根据这种拓扑序把出现次数计算出来, 同时, 我们也统计出了以某个节点为根的子树的出现次数总和.
有了每个状态的出现次数之后, 我们运行后缀自动机, 每次选择字典序最小的转移, 如果转移到达的状态所为根的子树的总出现次数大于k, 我们就可以输出这个字符, 同时k-=sum[trans[x][i]].
Code
#include <cstdio>
#include <cstring>
const int maxn = 5e5 + 1e2;
const int maxm = maxn << 1;
int N, T, k;
char str[maxn];
struct Suffix_Automaton {
int rt, last, trans[maxm][26], fa[maxm], sz, len[maxm], cnt[maxm];
int v[maxn], q[maxm], sum[maxm];
void init() {
sz = 0;
rt = last = ++sz;
}
void insert(int x) {
int p = last, np = last = ++sz;
len[np] = len[p] + 1;
cnt[np] = 1;
while (!trans[p][x] && p) {
trans[p][x] = np;
p = fa[p];
}
if (!p) {
fa[np] = 1;
} else {
int q = trans[p][x];
if (len[q] == len[p] + 1) {
fa[np] = q;
} else {
int nq = ++sz;
len[nq] = len[p] + 1;
memcpy(trans[nq], trans[q], sizeof(trans[q]));
fa[nq] = fa[q];
fa[q] = fa[np] = nq;
while (trans[p][x] == q) {
trans[p][x] = nq;
p = fa[p];
}
}
}
return;
}
void pre() {
for (int i = 1; i <= sz; i++)
v[len[i]]++;
for (int i = 1; i <= ::N; i++)
v[i] += v[i - 1];
for (int i = sz; i; i--)
q[v[len[i]]--] = i;
for (int i = sz; i; i--) {
int t = q[i];
if (T == 1)
cnt[fa[t]] += cnt[t];
else
cnt[t] = 1;
}
cnt[1] = 0;
for (int i = sz; i; i--) {
int t = q[i];
sum[t] = cnt[t];
for (int j = 0; j < 26; j++)
sum[t] += sum[trans[t][j]];
}
}
void dfs(int x, int k) {
if (k <= cnt[x])
return;
k -= cnt[x];
for (int i = 0; i < 26; i++)
if (int t = trans[x][i]) {
if (k <= sum[t]) {
putchar(i + 'a');
dfs(t, k);
return;
}
k -= sum[t];
}
}
} sam;
int main() {
#ifndef ONLINE_JUDGE
freopen("input", "r", stdin);
#endif
scanf("%s", str + 1);
N = strlen(str + 1);
scanf("%d %d", &T, &k);
sam.init();
for (int i = 1; i <= N; i++) {
sam.insert(str[i] - 'a');
}
sam.pre();
if (k > sam.sum[1])
printf("-1\n");
else
sam.dfs(1, k);
return 0;
}
[bzoj3998][TJOI2015]弦论-后缀自动机的更多相关文章
- 【BZOJ3998】[TJOI2015]弦论 后缀自动机
[BZOJ3998][TJOI2015]弦论 Description 对于一个给定长度为N的字符串,求它的第K小子串是什么. Input 第一行是一个仅由小写英文字母构成的字符串S 第二行为两个整数T ...
- 【bzoj3998】[TJOI2015]弦论 后缀自动机+dp
题目描述 对于一个给定长度为N的字符串,求它的第K小子串是什么. 输入 第一行是一个仅由小写英文字母构成的字符串S 第二行为两个整数T和K,T为0则表示不同位置的相同子串算作一个.T=1则表示不同位置 ...
- 【BZOJ-3998】弦论 后缀自动机
3998: [TJOI2015]弦论 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 2018 Solved: 662[Submit][Status] ...
- BZOJ 3998: [TJOI2015]弦论 [后缀自动机 DP]
3998: [TJOI2015]弦论 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 2152 Solved: 716[Submit][Status] ...
- BZOJ 3998: [TJOI2015]弦论 后缀自动机 后缀自动机求第k小子串
http://www.lydsy.com/JudgeOnline/problem.php?id=3998 后缀自动机应用的一个模板?需要对len进行一个排序之后再统计每个出现的数量,维护的是以该字符串 ...
- BZOJ 3998 TJOI2015 弦论 后缀自动机+DAG上的dp
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3998 题意概述:对于一个给定长度为N的字符串,求它的第K小子串是什么,T为0则表示不同位置 ...
- BZOJ 3998 [TJOI2015]弦论 ——后缀自动机
直接构建后缀自动机. 然后. 然后只需要再后缀自动机的go树上类似二分的方法进行查找即可,实际上是“26分”. 然后遇到了处理right集合的问题,然后觉得在go和parent树上上传都是可以的,毕竟 ...
- [TJOI2015]弦论(后缀自动机)
/* 一道在树上乱搞的题目 建立出parent树来, 然后就能搞出每个节点往后能扩展出几个串, 至于位置不同算同一个的话就强制让right集合大小为1即可 然后在树上类比权值线段树找第k大26分统计一 ...
- BZOJ.3998.[TJOI2015]弦论(后缀自动机)
题目链接 \(Description\) 给定字符串S,求其第K小子串.(若T=0,不同位置的相同子串算1个:否则算作多个) \(Solution\) 建SAM,处理出对于每个节点,它和它的所有后继包 ...
随机推荐
- telnet配置和telnet用法
搭建或配置网络环境时,经常会使用ping命令检查网络是否可达.有些时候Ping命令也不好使,比如因防火墙禁止或访问策略限制等.则可使用telnet测试映射端口或远程访问主机. Telnet协议是TCP ...
- mysql 两个时间段的差,可以是秒,天,星期,月份,年...
SELECT TIMESTAMPDIFF(SECOND, now(), "2012-11-11 00:00:00") 语法为:TIMESTAMPDIFF(unit,datetime ...
- Powerdesigner+Execel
1.将Powerdesigner中的表(PDM)导入到execel中 Ctrl+Shift+X/tool->Execute commands ->Edit/Run script 粘贴如下v ...
- CentOS 7安装Oracle 11gR2以及设置自启动
一.环境准备 1.正确无误的CentOS 7系统环境 CentOS 7安装:http://www.cnblogs.com/VoiceOfDreams/p/8043958.html 2.正确的JDK环境 ...
- JavaScript的三种类型检测typeof , instanceof , toString比较
1.typeof typeof是js的一个操作符,在类型检测中,几乎没有任何用处. typeof 返回一个表达式的数据类型的字符串,返回结果为javascript中的基本数据类型,包括:number. ...
- 使用SQL 提示优化sql
use index 在查询语句中表名的后面,添加use index来提供希望mysql去参考的索引列表,就可以让mysql不再考虑其他可用的索引 explain select * from renta ...
- Linux指令--ping
Linux系统的ping命令是常用的网络命令,它通常用来测试与目标主机的连通性,我们经常会说"ping一下某机器,看是不是开着".不能打开网页时会说"你先ping网关地址 ...
- web.xml 中<context-param>与<init-param>的区别与作用
<context-param>的作用: web.xml的配置中<context-param>配置作用 1. 启动一个WEB项目的时候,容器(如:Tomcat)会去读它的配置文件 ...
- struts2 action 页面与action参数的传递的三种方式
第一种: 初始页面: <form action="LoginAction.action" method="post"> 用户名:<input ...
- mybatis 中文文档
http://www.mybatis.org/mybatis-3/zh/sqlmap-xml.html