稍微学习了下第一次用后缀数组- - , 强行凑出答案 , 感觉现在最大的问题是很多算法都不知道 ,导致有的题一点头绪都没有(就像本题)。  /*推荐 《后缀数组——处理字符串的有力工具》——罗穗骞

后缀数组sa,  sa[ i ] = a表示字符串从第a个开始到结尾的字典序排序为i

本题是绕成了一个环,所以我将字符串重复一遍再用后缀数组模板跑一次。

顺序的话,取最后一个即可

反序的话,因为相同情况下,要求取得尽可能小。

hight求的是i和i-1的最长公共前缀,如果串i-1整个是公共最长前缀,即说明他们的字典序本该是一样的,只是在双倍的情况下i多算一部分,所以有先取i-1

//思路应该没问题,代码是在不停wa中莫名改出来的- -

#include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<string>
#include<map>
#include<set>
#include<vector>
#include<queue>
#include<stack>
using namespace std;
typedef long long ll; const int maxn=200000+10; int wa[maxn],wb[maxn],wv[maxn],wss[maxn],rak[maxn];
int cmp(int *r,int a,int b,int l)
{
return r[a]==r[b]&&r[a+l]==r[b+l];
}
void da(int *r,int *sa,int n,int m)
{
int i,j,p,*x=wa,*y=wb,*t;
for(i=0; i<m; i++) wss[i]=0;
for(i=0; i<n; i++) wss[x[i]=r[i]]++;
for(i=1; i<m; i++) wss[i]+=wss[i-1];
for(i=n-1; i>=0; i--) sa[--wss[x[i]]]=i;
for(j=1,p=1; p<n; j*=2,m=p)
{
for(p=0,i=n-j; i<n; i++) y[p++]=i;
for(i=0; i<n; i++) if(sa[i]>=j) y[p++]=sa[i]-j;
for(i=0; i<n; i++) wv[i]=x[y[i]];
for(i=0; i<m; i++) wss[i]=0;
for(i=0; i<n; i++) wss[wv[i]]++;
for(i=1; i<m; i++) wss[i]+=wss[i-1];
for(i=n-1; i>=0; i--) sa[--wss[wv[i]]]=y[i];
for(t=x,x=y,y=t,p=1,x[sa[0]]=0,i=1; i<n; i++)
x[sa[i]]=cmp(y,sa[i-1],sa[i],j)?p-1:p++;
}
return ;
} char s[2*maxn],rev[2*maxn];
int sa[maxn],r[maxn],height[maxn];
int n;
int p0,p1;
void build_height(int n)
{
int i,k=0;
for(i=0; i<n; i++)rak[sa[i]]=i;
for(i=0; i<n; i++)
{
if(k)k--;
int j=sa[rak[i]-1];
while(s[i+k]==s[j+k])k++;
height[rak[i]]=k;
}
} void solve(int x)
{
if(!x)
{
int t = 0;
memset(sa,0,sizeof(sa));
for(int i = 0; i < n; i++)
r[t++]=s[i]-'a'+1;
r[t++] = 0; da(r,sa,t,30);
p0 = sa[t-1]+1; }
else
{
int t = 0;
memset(sa,0,sizeof(sa));
for(int i = 0; i < n; i++)
r[t++]=rev[i]-'a'+1; r[t++] = 0;
da(r,sa,t,30);
build_height(t);
int k = t-2;
while(k>=4&&height[k]&&height[k]==n-sa[k-1])k--;
p1 = sa[k]; }
} int cmp(char*s,char*t,int p,int q)
{
for(int i=0; i<n; i++)
if(s[(p+i)%n]!=t[(q+i)%n])
return s[(p+i)%n]>t[(q+i)%n];
return 2;
} int main()
{
int T;
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
scanf("%s",s);
for(int i = 0; i < n; i++)
s[n+i] = s[i];
n = 2*n;
for(int i=0; i<n; i++)
rev[i]=s[n-1-i]; solve(0);
solve(1);
int ans,dir;
int tmp = p1;
// printf("%d %d\n",p0,tmp);
p1 = n-p1;
n/=2;
int d=cmp(s,rev,p0-1,tmp); //printf("%d %d\n",p0,tmp); if(d==2)
{
if(p0<=p1)ans=p0,dir=0;
else ans=p1,dir=1;
}
else if(d>0)ans=p0,dir=0;
else ans=p1,dir=1;
printf("%d %d\n",ans,dir);
}
return 0;
}

  

hdu 5442 (后缀数组)的更多相关文章

  1. hdu 3948 后缀数组

    The Number of Palindromes Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 262144/262144 K (J ...

  2. HDU - 3948 后缀数组+Manacher

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3948 题意:给定一个字符串,求字符串本质不同的回文子串个数. 思路:主要参考该篇解题报告 先按照man ...

  3. HDU 5769 后缀数组

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5769 [2016多校contest-4] 题意:给定一个字符,还有一个字符串,问这个字符串存在多少个不 ...

  4. hdu 2459 (后缀数组+RMQ)

    题意:让你求一个串中连续重复次数最多的串(不重叠),如果重复的次数一样多的话就输出字典序小的那一串. 分析:有一道比这个简单一些的题spoj 687, 假设一个长度为l的子串重复出现两次,那么它必然会 ...

  5. hdu 3518(后缀数组)

    题意:容易理解... 分析:这是我做的后缀数组第一题,做这个题只需要知道后缀数组中height数组代表的是什么就差不多会做了,height[i]表示排名第i的后缀与排名第i-1的后缀的最长公共前缀,然 ...

  6. HDU 5558 后缀数组

    思路: 这是一个错误的思路, 因为数据水才过= = 首先求出来后缀数组 把rank插到set里 每回差i两边离i近的rank值,更新 如果LCP相同,暴力左(右)继续更新sa的最小值 //By Sir ...

  7. HDU 4691 后缀数组+RMQ

    思路: 求一发后缀数组,求个LCP 就好了 注意数字有可能不只一位 (样例2) //By SiriusRen #include <bits/stdc++.h> using namespac ...

  8. K-th occurrence HDU - 6704 (后缀数组+二分线段树+主席树)

    大意: 给定串s, q个询问(l,r,k), 求子串s[l,r]的第kk次出现位置. 这是一篇很好的题解: https://blog.csdn.net/sdauguanweihong/article/ ...

  9. Hdu 1403(后缀数组)

    题目链接 Longest Common Substring Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/32768 K ...

随机推荐

  1. 201621123043 《Java程序设计》第7周学习总结

    1. 本周学习总结 2.书面作业 1. GUI中的事件处理 1.1 写出事件处理模型中最重要的几个关键词. 事件:用户的操作. 事件源:产生事件的组件. 事件监听程序:对事件进行处理的操作所引发的相关 ...

  2. JAVA使用和操作properties文件

    java中的properties文件是一种配置文件,主要用于表达配置信息,文件类型为*.properties,格式为文本文件,文件的内容是格式是"键=值"的格式,在properti ...

  3. OO面向对象课程作业1-3总结

    作业一.多项式的加减运算 1.设计要点与自我分析 我设计的类图 老师建议类图 我设计了两个类来进行多项式的计算,类Polynomial进行多项式的存储和输入输出,第二个类进行多项式加减运算.而加减运算 ...

  4. CentOS7 防火墙firewalld详细操作

    1.firewalld的基本使用 启动: systemctl start firewalld 查看状态: systemctl status firewalld  停止: systemctl disab ...

  5. RocketMQ(二):RPC通讯

    匠心零度 转载请注明原创出处,谢谢! RocketMQ网络部署图 NameServer:在系统中是做命名服务,更新和发现 broker服务. Broker-Master:broker 消息主机服务器. ...

  6. kubernetes入门(09)kubernetes1.7集群安装(2017/11/13)

    CentOS7.3利用kubeadm安装kubernetes1.7.3完整版(官方文档填坑篇) https://www.cnblogs.com/liangDream/p/7358847.html 一. ...

  7. GIT入门笔记(10)- 多种撤销修改场景和对策

    场景1:当你改乱了工作区某个文件的内容,想直接丢弃工作区的修改时,用命令git checkout -- file. 场景2:当你不但改乱了工作区某个文件的内容,还添加到了暂存区时,想丢弃修改,分两步, ...

  8. java实现两个int数交换

    普通方法,进阶方法,大神方法 @Test public void test3(){ int m = 5; int n = 12; //要求m和n交换位置 System.out.println(&quo ...

  9. Object.defineProperties()和Object.defineProperty()方法

    Object.defineProperty() 方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性, 并返回这个对象. 语法:Object.defineProperty(obj, pro ...

  10. linux环境 安装chromedriver 和 phantomjs的方法

    1 首先要下载浏览器驱动: 常用的是chromedriver 和phantomjs chromedirver下载地址: https://npm.taobao.org/mirrors/chromedri ...