MANACHER---求最长回文串
求最长回文串,如果是暴力的方法的话,会枚举每个字符为中心,然后向两边检测求出最长的回文串,时间复杂度在最坏的情况下就是0(n^2),为什么时间复杂度会这么高,因为对于每一个作为中心的字符的检测是独立的,没有充分利用前面比较过信息,这就类似暴力求字符串的匹配最糟糕的情况下是0(n*m),然后通过预处理的信息把时间复杂度降低也就是kmp算法;
MANACHER算法:
先假设所有回文串都是以某个字符为中心的,即回文串的长度都是奇数;
lc[ i ]保存的是以位置i的字符为中心的最长回文串到最右边的距离
先假设以知lc[i],(0<=i<x)求lc[x]
设p=k+lc[k]-1,k是使p最大的i的取值,如下图
如果x>p那么直接以x为中心进行检测,并更新k;
如果x<=p那么对于以k为中心x的对称点就是j,并且lc[j]的值已经知道了,
If (lc[j]<p-x+1) lc[x]=lc[j]
因为s[j-1]!=s[j+1], s[j-1]=s[x-1],s[j+1]=s[x+1],所以s[x-1]!=s[x+1]如图:
If (lc[j]>=p-x+1) 那么lc[x]至少是lc[j] 如图:
但对于蓝色位置是不是还需要检测
时间复杂度分析:因为对于每一位s[i]都只被检测了一次,所以是o(n);
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
现在回到最原先的问题,因为回文串有可能是“1221”这种长度是偶数的,MANACHER算法提供了一个构造的方法可以统一这两种情况:在两个字符之间插入一个没有出现过的字符,如‘#’,那么”1221”->”#1#2#2#1#”这样所有的回文串长度都是奇数的了,为了处理方便再在该字符串前面加一个‘$’字符,这样字符串就是$#1#2#2#1#
求出lc[i]后,对于以s[i]为中心的字符串,如果s[i]==’#’,那么len=(lc[i]-1)/2*2,因为lc[i]-1肯定是偶数(因为该回文串的最左和最右肯定是’#’)所以len=lc[i]-1;
如果s[i]!=’#’,那么len=(lc[i] - 1)/2*2+1;因为lc[i]-1肯定是奇数,所以len=lc[i]-2+1=lc[i]-1;
所以最后的解就是最大lc[i]-1;
void Manacher(char *s){
s1[]='$';
int nn=strlen(s),c=;
for (int i=;i<nn;i++){
s1[c++]='#';
s1[c++]=s[i];
}s1[c++]='#';s1[c++]='\0';
// cout<<s<<endl<<s1<<endl;
lc[]=;lc[]=;
int k=, p, j;
for (int i=;i<c;i++){
p=k+lc[k]-;
if (p<i){
int t=;
while (i-t>= && i+t<c && s1[i-t]==s1[i+t]) t++;
lc[i]=t; k=i;
}else {
j=*k-i;
if (lc[j]<p-i+) lc[i]=lc[j];
else {
int t=p-i+;
while (i-t>= && i+t<c && s1[i-t]==s1[i+t]) t++;
lc[i]=t; k=i;
}
}
}
int ret=;
for (int i=;i<c;i++){
//cout<<lc[i]<<" ";
if (lc[i]->ret) ret=lc[i]-;
}//cout<<endl;
printf("%d\n",ret);
}
MANACHER---求最长回文串的更多相关文章
- Manacher算法 - 求最长回文串的利器
求最长回文串的利器 - Manacher算法 Manacher主要是用来求某个字符串的最长回文子串. 不要被manacher这个名字吓倒了,其实manacher算法很简单,也很容易理解,程序短,时间复 ...
- Manacher's Algorithm 马拉车算法(求最长回文串)
作用:求一个字符串中的最长子串,同时还可以求所有子串的长度. 题目链接: https://vjudge.net/contest/254692#problem/B 最长回文串长度的代码: int Man ...
- hdu 3068 最长回文 (Manacher算法求最长回文串)
参考博客:Manacher算法--O(n)回文子串算法 - xuanflyer - 博客频道 - CSDN.NET 从队友那里听来的一个算法,O(N)求得每个中心延伸的回文长度.这个算法好像比较偏门, ...
- hdu 3068 最长回文 【Manacher求最长回文子串,模板题】
欢迎关注__Xiong的博客: http://blog.csdn.net/acmore_xiong?viewmode=list 最长回文 ...
- Manacher(最长回文串)
http://acm.hdu.edu.cn/showproblem.php?pid=3068 最长回文 Problem Description 给出一个只由小写英文字符a,b,c...y,z组成的字符 ...
- Manacher算法,最长回文串
给你10000长度字符串,然你求最长回文字串,输出长度,暴力算法肯定超时 #include <iostream> #include <string> #include < ...
- Manacher 计算最长回文串
转自 http://blog.sina.com.cn/s/blog_3fe961ae0101iwc2.html 寻找字符串中的回文,有特定的算法来解决,也是本文的主题:Manacher算法,其时间复杂 ...
- Manacher 求最长回文子串算法
Manacher算法,是由一个叫Manacher的人在1975年发明的,可以在$O(n)$的时间复杂度里求出一个字符串中的最长回文子串. 例如这两个回文串“level”.“noon”,Manacher ...
- POJ 3376 Finding Palindromes(manacher求前后缀回文串+trie)
题目链接:http://poj.org/problem?id=3376 题目大意:给你n个字符串,这n个字符串可以两两组合形成n*n个字符串,求这些字符串中有几个是回文串. 解题思路:思路参考了这里: ...
- Manacher求最长回文
#1032 : 最长回文子串 时间限制:1000ms 单点时限:1000ms 内存限制:64MB 描写叙述 小Hi和小Ho是一对好朋友.出生在信息化社会的他们对编程产生了莫大的兴趣,他们约定好互相帮助 ...
随机推荐
- ORACLE将表中的数据恢复到某一个时间点
执行如下SQL将test_temp表中的数据恢复到 2013-04-26 21:06:00 注意,这里一定要先删除全部数据,否则可能会导致数据重复 delete from test_temp; in ...
- Android开发之TextView实现跑马灯效果
TextView及其子类,当字符内容太长显示不下时可以省略号代替未显示的字符:省略号可以在显示区域的起始,中间,结束位置,或者以跑马灯的方式显示文字(textview的状态为被选中). 其实现只需在x ...
- C++中关于指针初始化和使用NULL的理解
1.严禁使用未被初始化的指针:C++创建指针的时候,只分配存储地址的内存,并不会分配存储数据的内存,所以指针可能指向任何位置. (1)使用解除运算符(*)之前,一定要对指针初始化,否则若声明的指针刚好 ...
- 面试题_82_to_87_Date、Time 及 Calendar 的面试题
82)在多线程环境下,SimpleDateFormat 是线程安全的吗?(答案)不是,非常不幸,DateFormat 的所有实现,包括 SimpleDateFormat 都不是线程安全的,因此你不应该 ...
- 异常:Caused by: java.lang.NoClassDefFoundError: Could not initialize class net.sf.log4jdbc.Properties
参考文章: 使用Log4jdbc-log4j2监听MyBatis中运行的SQL和Connection 使用 log4jdbc格式化输出SQL,maven配置如下: <dependency> ...
- Linux 下查看文件字符编码和转换编码
Linux 下查看文件字符编码和转换编码 如果你需要在Linux中操作windows下的文件,那么你可能会经常遇到文件编码转换的问题.Windows中默认的文件格式是GBK(gb2312),而Linu ...
- 使用stringstream时的清空操作
在C++中可以使用stringstream来很方便的进行类型转换,字符串串接,不过注意重复使用同一个stringstream对象时要先继续清空,而清空很容易想到是clear方法,而在stringstr ...
- bzoj2792
首先想到二分答案是吧,设为lim 这道题难在判定,我们先不管将一个数变为0的条件 先使序列满足相邻差<=lim,这个正着扫一遍反着扫一遍即可 然后我们就要处理将一个数变为0的修改代价 当i变为0 ...
- git - svn 平滑到 git
1. 建立自己的git仓库,需要是空git仓库 2. checkout 你的 git仓库 3. svn忽略.git文件,忽略.git .gitignore 4. 把 .git文件拷到你的 svn仓库 ...
- liunx之zip格式的解压命令
zip -r myfile.zip ./* 将当前目录下的所有文件和文件夹全部压缩成myfile.zip文件,-r表示递归压缩子目录下所有文件. 2.unzip unzip -o -d /home/s ...