当初合肥区域赛的题(现场赛改了数据范围就暴力过了),可惜当初后缀数组算法的名字都没听过,现在重做下。

i从1到n - 1,每次枚举rank[i]附近的排名,并记录当起点小于i时的LCP(rank[i], d)的最大值,或是在LCP不变时更新起点。

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cstdlib>
using namespace std;
const int N = 110008, INF = 0x3F3F3F3F; int val[N], ws0[N], wa[N], wb[N];
int sa[N], rk[N], height[N]; char str[N]; bool cmp(int str[], int a, int b, int l){
return str[a] == str[b] && str[a + l] == str[b + l];
} void da(char str[], int n, int m){
int *x = wa, *y = wb; memset(ws0, 0, sizeof(ws0));
for(int i = 0; i < n; i++)
ws0[x[i] = str[i]]++; for(int i = 1; i < m; i++)
ws0[i] += ws0[i - 1]; for(int i = n - 1; i >= 0; i--)
sa[--ws0[ x[i]] ] = i; for(int j = 1, p = 1; p < n; j *= 2, m = p){
p = 0;
for(int i = n - j; i < n; i++)
y[p++] = i;
for(int i = 0; i < n; i++)
if(sa[i] >= j)
y[p++] = sa[i] - j;
for(int i = 0; i < n; i++)
val[i] = x[ y[i] ]; memset(ws0, 0, sizeof(ws0));
for(int i = 0; i < n; i++)
ws0[val[i]]++; for(int i = 1; i < m; i++)
ws0[i] += ws0[i - 1]; for(int i = n - 1; i >= 0; i--)
sa[--ws0[ val[i] ]] = y[i]; swap(x, y);
x[sa[0]] = 0;
p = 1;
for(int i = 1; i < n; i++)x[sa[i]] = cmp(y, sa[i - 1], sa[i], j)? p - 1:p++;
}
} void calHeight(char str[], int n){
for(int i = 1; i <= n; i++){
rk[ sa[i] ] = i;
}
int k = 0;
for(int i = 0; i < n; i++){
if(k){
k--;
}
int j = sa[rk[i] - 1];
while(str[i + k] == str[j + k]){
k++;
}
height[rk[i]] = k;
}
} int rmq[N][20], tab[N]; void initaskRMQ(int n){
tab[1] = 0;
for(int i = 2; i <= n; i++){
tab[i] = ( (i & (i - 1)) == 0)? tab[i - 1] + 1:tab[i - 1];
}
for(int i = 1; i <= n; i++){
rmq[i][0] = height[i];
}
for(int i = 1; i <= tab[n]; i++){
for(int j = 1; j <= n + 1 - (1 << i); j++){
int a = rmq[j][i - 1];
int b = rmq[j + (1 <<(i - 1))][i - 1];
rmq[j][i] = min(a, b);
}
}
} int askRMQ(int a, int b){
int k = tab[b -a + 1];
b -= (1 << k ) - 1;
return min(rmq[a][k], rmq[b][k]);
} int LCP(int a, int b){
if(a > b){
swap(a, b);
}
return askRMQ(a + 1, b);
} int main(){
int t;
cin>>t;
for(int cas = 1; cas <= t; cas++){
scanf("%s", str);
int n = strlen(str);
da(str, n + 1, 'z' + 1);
calHeight(str, n); initaskRMQ(n);
printf("Case #%d:\n", cas);
printf("%d %d\n", -1, str[0]);
for(int i = 1; i < n; ){
int st = INF, k = 1;
int d = rk[i] + 1;
int tp;
while(d <= n && (tp = LCP(d, rk[i])) >= k){
if(sa[d] < i){
if(tp > k){
k = tp;
st = sa[d];
}else{
if(sa[d] < st){
st = sa[d];
}
}
}
d++;
} d = rk[i] - 1;
while(d >= 1 && (tp = LCP(d, rk[i])) >= k){
if(sa[d] < i){
if(tp > k){
k = tp;
st = sa[d];
}else{
if(sa[d] < st){
st = sa[d];
}
}
}
d--;
}
if(st < i){
printf("%d %d\n", k, st);
i += k;
}else{
printf("%d %d\n", -1, (int)str[i]);
i++;
}
}
}
return 0;
}

  加油吧! fighting!!!

HDU5558 Alice's Classified Message(合肥区域赛 后缀数组)的更多相关文章

  1. hdu5558 Alice's Classified Message

    地址:http://acm.split.hdu.edu.cn/showproblem.php?pid=5558 题目: Alice's Classified Message Time Limit: 1 ...

  2. (HDU 5558) 2015ACM/ICPC亚洲区合肥站---Alice's Classified Message(后缀数组)

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=5558 Problem Description Alice wants to send a classi ...

  3. HDU5556 Land of Farms(二分图 2015 合肥区域赛)

    容易想到将问题转化为求图的独立数问题 ,但求一般图的独立集是一个NPC问题,需要一些转化. 状态压缩,枚举每个上古农场是否选择,然后将剩下的新农场根据i + j奇偶性分为x , y集. 结果为 max ...

  4. HUID 5558 Alice's Classified Message 后缀数组+单调栈+二分

    http://acm.hdu.edu.cn/showproblem.php?pid=5558 对于每个后缀suffix(i),想要在前面i - 1个suffix中找到一个pos,使得LCP最大.这样做 ...

  5. 【HDOJ5558】Alice's Classified Message(后缀数组)

    题意:给定一个长度为n的下标从0开始的小写字母字符串,每次对于当前的i寻找一个j使得后缀i与后缀j的lcp最大,(j<i)若lcp相同则取较小的 若lcp为0则输出0与当前字符,i自增1,否则输 ...

  6. Alice's Classified Message HDU - 5558 后缀自动机求某个后缀出现的最早位置

    题意: 给定一个长度不超过 10W 的只包含小写字母的字符串,从下标 0 到 n−1.从下标 0 开始操作, 每次对于下标 pos查找下标 pos 开始的子串中最长的在其他地方出现过的长度,其他出现的 ...

  7. HDU 5558 Alice's Classified Message(后缀数组+二分+rmq(+线段树?))

    题意 大概就是给你一个串,对于每个\(i\),在\([1,i-1]\)中找到一个\(j\),使得\(lcp(i,j)\)最长,若有多个最大\(j\)选最小,求\(j\)和这个\(lcp\)长度 思路 ...

  8. POJ.2774.Long Long Message/SPOJ.1811.LCS(后缀数组 倍增)

    题目链接 POJ2774 SPOJ1811 LCS - Longest Common Substring 比后缀自动机慢好多(废话→_→). \(Description\) 求两个字符串最长公共子串 ...

  9. HDU 6194 string string string 2017沈阳网络赛 后缀数组

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6194 题意:告诉你一个字符串和k , 求这个字符串中有多少不同的子串恰好出现了k 次. 解法:后缀数组 ...

随机推荐

  1. HTTP 格式

    HTTP请求报文和HTTP响应报文 HTTP报文是面向文本的,报文中的每一个字段都是一些ASCII码串,各个字段的长度是不确定的.HTTP有两类报文:请求报文和响应报文. HTTP请求报文 一个HTT ...

  2. Game of Life I & II

    According to the Wikipedia's article: "The Game of Life, also known simply as Life, is a cellul ...

  3. ios7 上 UIActivity 用的image

    在ios8 上UIActivityCategoryShare类型的UIActivity的图标支持彩色图片了,但是在ios7上不行,ios8上的 UIActivityCategoryAction类型也不 ...

  4. Kafka单机环境部署

    前面说过Kafka集群环境的部署,现在主要说一下在本地测试中Kafka单机环境的部署,和前面一样首先保证zookeeper服务的正常运行,然后解压并释放kafka安装包,并放到指定位置: tar -x ...

  5. Windows下安装node

    1.安装node及npm Windows下安装软件都是傻瓜式安装,首先登陆官网(https://nodejs.org/en/)下载对应的node程序,然后双击进行安装.安装过程基本上是点击'Next' ...

  6. Python缩小图像

    LyncLynn用途: 缩小图像 # -*- coding: UTF-8 -*- #Version: V1.0 #Author:lynclynn #CreateDate:20151201 #Updat ...

  7. ATS(App Transport Security)对HTTP协议屏蔽引起的问题

    一.问题描述 在学习网络处理的过程,发现代码都没错,运行时会收到如下错误提示: App Transport Security has blocked a cleartext HTTP (http:// ...

  8. HTML DOM scale() 方法

    语法 scale(sx, sy) 参数 参数 描述 sx, sy 水平和垂直的缩放因子. 描述 scale() 方法为画布的当前变换矩阵添加一个缩放变换.缩放通过独立的水平和垂直缩放因子来完成.例如, ...

  9. js事件监听器用法实例详解

    这篇文章主要介绍了js事件监听器用法,以实例形式较为详细的分析了javascript事件监听器使用注意事项与相关技巧,需要的朋友可以参考下本文实例讲述了js事件监听器用法.分享给大家供大家参考.具体分 ...

  10. 51nod 1117 聪明的木匠 (哈夫曼树)

    题目:传送门. 题意:中文题. 题解:就是构造一颗哈夫曼树,数据结构里的知识. #include <iostream> #include <cstdio> #include & ...