【BZOJ2320】最多重复子串 调和级数+hash
【BZOJ2320】最多重复子串
Description
一个字符串P的重复数定义为最大的整数R,使得P可以分为R段连续且相同的子串。比方说,“ababab”的重复数为3,“ababa”的重复数为1。
Your Task
对于给定的串S,找出S的一个子串K使得K的重复数最大。
Input
第一行T表示数据组数
对于每组数据,一行中一个仅包含小写字母的字符串S
Output
对于每组数据,在一行中输出K,如果有多个解,输出字典序最小的那一个
Sample Input
ccabababc
daabbccaa
Sample Output
aa
HINT
100%:T≤10,S的长度不超过100000
题解:居然能第二次遇到这种套路,真是不容易(第一次在股市的预测)。
如果重复数为1,则答案就是最小的字符,下面只考虑重复数不是1的情况。
回忆next数组的性质,一个串的最小循环节为n-next[n](如果有的话),而我们要枚举什么呢?我们枚举的就是0~n-next[n]以及next[n]~n的部分。
具体地,我们枚举循环节的长度为len,然后每隔len的长度设一个关键点,这样就保证了每个循环节都只包含一个关键点。然后我们对于每个关键点i,求出它和下一个关键点的最长公共前缀A和最长公共后缀B,用$\lfloor{A+B-1\over len}\rfloor+1$更新答案。
如何求字典序最小呢?因为极长重复串的个数是O(n)的(Claris说的。。。),所以暴力判断即可。
时间复杂度取决于如何求LCP和LCS,如果用hash+二分的话是$O(nlog^2n)$的。
#include <cstdio>
#include <iostream>
#include <cstring>
using namespace std;
typedef unsigned long long ull;
const int maxn=100010;
int n,ans,ap,bp;
ull hs[maxn],bs[maxn];
char str[maxn];
inline ull hash(int a,int b) {return hs[b]-((!a)?0:hs[a-1]*bs[b-a+1]);}
inline int lcs(int a,int b)
{
int l=0,r=min(a,b)+2,mid;
while(l<r)
{
mid=(l+r)>>1;
if(hash(a-mid+1,a)==hash(b-mid+1,b)) l=mid+1;
else r=mid;
}
return l-1;
}
inline int lcp(int a,int b)
{
int l=0,r=n-max(a,b)+1,mid;
while(l<r)
{
mid=(l+r)>>1;
if(hash(a,a+mid-1)==hash(b,b+mid-1)) l=mid+1;
else r=mid;
}
return l-1;
}
void updata(int a,int b)
{
if(ap==-1) ap=a,bp=b;
else
{
int l1=b-a+1,l0=bp-ap+1,k=min(lcp(a,ap),min(l1,l0));
if((k<=l1?str[a+k]:0)<(k<=l0?str[ap+k]:0)) ap=a,bp=b;
}
}
inline void calc(int x)
{
for(int i=0,a,b,c;i+x<n;i+=x) if(str[i]==str[i+x])
{
a=lcs(i,i+x),b=lcp(i,i+x),c=(a+b-1)/x+1;
if(c>ans) ans=c,ap=bp=-1;
if(ans!=1&&c==ans) for(int j=i-a+1;j+c*x-1<=i+x+b-1;j++) updata(j,j+c*x-1);
}
}
void work()
{
scanf("%s",str),n=strlen(str),ans=1,ap=bp=-1;
int i;
for(bs[0]=1,i=1;i<=n;i++) bs[i]=bs[i-1]*131;
for(i=0;i<n;i++) hs[i]=((!i)?0:hs[i-1])*131+str[i],ap=(ap==-1||str[i]<str[ap])?i:ap,bp=ap;
for(i=1;i<=n&&n/i>=ans;i++) calc(i);
for(i=ap;i<=bp;i++) printf("%c",str[i]);
printf("\n");
}
int main()
{
int T; scanf("%d",&T);
while(T--) work();
return 0;
}//1 ababacac
【BZOJ2320】最多重复子串 调和级数+hash的更多相关文章
- BZOJ2320 : 最多重复子串
本题就是求重复数最多的字典序最小的$runs$,如果重复数为1,那么做法显然,然后只考虑重复数大于1的情况. 从小到大枚举长度$len$,对于每个关键点$x=i\times len$,有且仅有一个长度 ...
- 【TOJ 2406】Power Strings(KMP找最多重复子串)
描述 Given two strings a and b we define a*b to be their concatenation. For example, if a = "abc& ...
- 【POJ 3693】Maximum repetition substring 重复次数最多的连续重复子串
后缀数组的论文里的例题,论文里的题解并没有看懂,,, 求一个重复次数最多的连续重复子串,又因为要找最靠前的,所以扫的时候记录最大的重复次数为$ans$,扫完后再后从头暴力扫到尾找重复次数为$ans$的 ...
- spoj687 后缀数组重复次数最多的连续重复子串
REPEATS - Repeats no tags A string s is called an (k,l)-repeat if s is obtained by concatenating k& ...
- POJ-3693-Maximum repetition substring(后缀数组-重复次数最多的连续重复子串)
题意: 给出一个串,求重复次数最多的连续重复子串 分析: 比较容易理解的部分就是枚举长度为L,然后看长度为L的字符串最多连续出现几次. 既然长度为L的串重复出现,那么str[0],str[l],str ...
- poj 3693 后缀数组 重复次数最多的连续重复子串
Maximum repetition substring Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 8669 Acc ...
- POJ-3693/HDU-2459 Maximum repetition substring 最多重复次数的子串(需要输出具体子串,按字典序)
http://acm.hdu.edu.cn/showproblem.php?pid=2459 之前hihocoder那题可以算出最多重复次数,但是没有输出子串.一开始以为只要基于那个,每次更新答案的时 ...
- Repeats SPOJ - REPEATS(重复次数最多的连续重复子串)
论文题例8 https://blog.csdn.net/queuelovestack/article/details/53031731这个解释很好 其实,当枚举的重复子串长度为i时,我们在枚举r[i* ...
- POJ 3693 Maximum repetition substring(最多重复次数的子串)
Maximum repetition substring Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 10461 Ac ...
随机推荐
- 持续集成之代码质量管理-Sonar
原文:http://blog.csdn.net/abcdocker/article/details/53840582 Sonar介绍 Sonar 是一个用于代码质量管理的开放平台.通过插件机制,Son ...
- poj 1426 Find The Multiple (bfs 搜索)
Find The Multiple Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 18012 Accepted: 729 ...
- 【android相关】【问题解决】R.java文件丢失
在进行android开发过程中,有时候,我们会遇到gen文件中R.java丢失的现象.重新build,或者clean工程,close并重新打开Project,但有时也没解决. 这可能是由于不小心把xm ...
- zh-cn en-uk表示语言(文化)代码与国家地区对照表
af 公用荷兰语 af-ZA 公用荷兰语 - 南非 sq 阿尔巴尼亚 sq-AL 阿尔巴尼亚 -阿尔巴尼亚 ar 阿拉伯语 ar-DZ 阿拉伯语 -阿尔及利亚 ar-BH 阿拉伯语 -巴林 ar-EG ...
- iOS Core ML与Vision初识
代码地址如下:http://www.demodashi.com/demo/11715.html 教之道 贵以专 昔孟母 择邻处 子不学 断机杼 随着苹果新品iPhone x的发布,正式版iOS 11也 ...
- AIDL调用指南
近期有需求要实现两个apk之间的通信,想到用AIDL来实现,现写一个demo学习下AIDL怎样使用. 这里我要实现一个apk(client端)调用还有一个apk(server端)的方法. 先实现ser ...
- Go环境IDE安装配置
终于配好了自己的Go环境,每天可以来一点积累了. MAC安装配置过程参考了如下几个博文~谢谢 Intellij安装配置: http://blog.csdn.net/fenglailea/article ...
- Webpack 的 HtmlWebpackPlugin 如何控制某个 chunks 的 inject 位置?
https://segmentfault.com/q/1010000006591131 通过修改 HtmlWebpackPlugin 源码实现了 修改后的配置: new HtmlWebpackPlug ...
- sshd服务安装和配置管理
1.SSHD简介(介绍) SSH协议:安全外壳协议,为Secure Shell的缩写,SSH为建立在应用层和传输层基础上的安全协议. sshd服务使用SSH协议可以用来进行远程控制,或在计算机之间传送 ...
- robot.txt 文件 作用和语法
seo工作者应该不陌生,robots.txt文件是每一个搜索引擎蜘蛛到你的网站之后要寻找和访问的第一个文件,robots.txt是你对搜索引擎制定的一个如何索引你的网站的规则.通过该文件,搜索引擎就可 ...