回文串指给定的字符串,正着读和反着读都是一样的。如ADA,反过来还是ADA即为回文串。最长回文子串指查找一给定字符串中最长的回文串。
通常有以下4种解法。主要考虑的是时间复杂度。
1:穷举法
穷举所有的子串,找出是回文串的子串,统计出最长的一个。
求每一个子串时间复杂度O(N^2),判断子串是不是回文O(N),两者是相乘关系,所以时间复杂度为O(N^3)。
代码如下:
- #include <iostream>
- #include <string>
- using namespace std;
-
-
-
-
- bool checkPalindrome(const string &s, int i, int j){
- for(int k = 0; k < (j-i+1)/2; k++){
- if(s[i+k] != s[j-k]) return false;
- }
- return true;
- }
-
- void longestPadEnum(const string &s){
-
- int begin=0;
- int maxSize = 0;
- for(int i = 0; i < s.size(); i++){
- for(int j = i+1; j < s.size(); j++){
- if(checkPalindrome(s, i, j) && j-i > maxSize){
- begin = i;
- maxSize = j-i;
- }
- }
- }
- cout << s.substr(begin, maxSize+1) << endl;
- }
-
- int main(){
- string s;
- while(cin >> s){
- longestPadEnum(s);
- }
- return 0;
- }
2:中心扩展法
回文串都是从中心开始的,我们把字符串的每个字母当做中心,向两边扩展,这样找最长的回文串。时间复杂度就变为了O(N^2)。
但是对于每个字母当做中心进行扩展的情况,都要考虑此时的回文串是偶数还是奇数,然后找出最长的一个。
如:(1)像aba,这样的长度为奇数。对应着代码标注的1:奇数情况
(2)像abba这样长度为偶数的回文串。处理对应着标注的2:偶数情况
代码如下:
-
-
-
-
-
- int getPalindrome1(const string &s, int i){
- int j = 0;
- int maxSize = 0;
- while(i-j >= 0 && i+j < s.size()){
- if(s[i-j] == s[i+j]) j++;
- else break;
- }
- if(2*j-1 > maxSize){
- maxSize = 2*j-1;
- }
- j = 0;
- while(i-j >= 0 && i+j+1 < s.size()){
- if(s[i-j] == s[i+j+1])j++;
- else break;
- }
- if(2*j > maxSize){
- maxSize = 2*j;
- }
- return maxSize;
- }
-
- void longestPadExtend1(const string &s){
-
- int maxSize = 0;
- int mid = 0;
- for(int i = 0; i < s.size(); i++){
- int padLen = getPalindrome1(s, i);
- if(padLen > maxSize){
- maxSize = padLen;
- mid = i;
- }
- }
-
-
- int begin = 0;
- if(maxSize % 2)begin = mid- maxSize/2;
- else begin = mid - maxSize/2 + 1;
- int end = mid + maxSize/2;
- cout << s.substr(begin, end+1) << endl;
- }
当然存在这另外一种解决回文串是奇数还是偶数的问题。此时可以在字符串的头尾及每两个字符之间加入新的字符#(假设字符#不在字符串中出现过)。此时以每个字符向两边扩展,我们会发现#向后面扩展的长度都为奇数,字符串中字符向后扩展的长度都为偶数(包括自己)。如果是#向后扩展了3个,如为b#a#a#c,此时回文串为2;如果是字符串中的字符向后扩展了4个,如为d#b#a#b#c,,则回文串为3;因此可指此时最大的回文串个数就是某点字符向后扩展的个数-1。这其实就是manacher算法中的第一个性质。
代码如下:
-
- int getPalindrome2(const string &s, int i){
- int j = 0;
- int maxSize = 0;
- while(i-j >= 0 && i+j < s.size()){
- if(s[i-j]==s[i+j])j++;
- else break;
- }
- if(j > maxSize)
- maxSize = j;
- return maxSize;
- }
-
-
-
- void longestPadExtend2(const string &s){
- string str = "#";
- for(int i = 0; i < s.size(); i++)
- {
- str += s[i];
- str += "#";
- }
- int maxSize = 0;
- int mid = 0;
- for(int i = 0; i < str.size(); i++){
- int pLen = getPalindrome2(str, i);
- if(pLen > maxSize){
- maxSize = pLen;
- mid = i;
- }
- }
-
- int begin = 1, end = 1;
- if(maxSize % 2){
- begin = mid - maxSize + 1;
- end = mid + maxSize - 2;
- }else {
- begin = mid - maxSize +2;
- end = mid + maxSize - 2;
- }
- for(int j = begin; j <= end; j+=2)
- cout << str[j];
- cout << endl;
-
-
- }
3:动态规划
我们用c[i][j]=1来表示字符串从i到j为回文串,用c[i][j]=0来表示字符串从i到j为非回文串。因此当c[i][j]为回文串,c[i+1][j-1]也必为回文串。此时j-i+1即为最长的回文子串。
依据动态规划的步骤:
初始化:c[i][i]=1; 如果s[i]==s[i+1],则c[i][i+1]=1.
时间复杂度:O(N^2),但是此时比中心扩展法需要额外的O(N^2)空间
代码如下:
-
-
-
-
- #include <iostream>
- #include <string>
- using namespace std;
- #define MAXSIZE 100
-
- void findDPLongestPad(string s){
-
- int c[MAXSIZE][MAXSIZE];
- memset(c, 0, sizeof(c));
-
- int maxSize = 1;
- int begin = 0;
- for(int i = 0; i < s.size(); i++)
- {
- c[i][i] = 1;
- if(i+1 < s.size() && s[i] == s[i+1]){
- c[i][i+1] = 1;
- maxSize = 2;
- begin = i;
- }
- }
-
- for(int len = 3; len <= s.size(); len++){
- for(int i = 0; i <= s.size() - len; i++ ){
- int j = i+len-1;
- if(s[i] == s[j] && c[i+1][j-1]){
- c[i][j] = 1;
- maxSize = len;
- begin = i;
- }
- }
- }
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- for(int i = 0; i < s.size(); i++)
- {
- for(int j = 0; j < s.size(); j++)
- cout << c[i][j] << " ";
- cout << endl;
-
- }
-
- cout << s.substr(begin, begin+maxSize) << endl;
- }
-
-
- int main(){
- string s;
- while(cin >> s){
- findDPLongestPad(s);
- }
- return 0;
- }
4:Manacher算法—O(n)
(1)算法基本要点:首先用一个非常巧妙的方式,将所有可能的奇数/偶数长度的回文子串都转换成了奇数长度:在每个字符的两边都插入一个特殊的符号。比如 abba 变成 #a#b#b#a#, aba变成 #a#b#a#。 为了进一步减少编码的复杂度,可以在字符串的开始加入另一个特殊字符,这样就不用特殊处理越界问题,比如$#a#b#a#。
(2)例子:
下面以字符串12212321为例,经过上一步,变成了 S[] = "$#1#2#2#1#2#3#2#1#";
然后用一个数组 P[i] 来记录以字符S[i]为中心的最长回文子串向左/右扩张的长度(包括S[i]),比如S和P的对应关系:
S # 1 # 2 # 2 # 1 # 2 # 3 # 2 # 1 #
P 1 2 1 2 5 2 1 4 1 2 1 6 1 2 1 2 1
(p.s. 可以看出,P[i]-1正好是原字符串中回文串的总长度)——性质1
(3)计算P[i]
下面就是要计算p[i],该算法增加两个辅助变量id和mx,其中id表示最大回文子串中心的位置,mx则为id+P[id],也就是最大回文子串的边界。
这个算法的关键点就在这里了:如果mx > i,那么P[i] >=MIN(P[2 * id - i], mx - i)。
代码:
- if(mx > i)
- {
- p[i] = (p[2*id - i] < (mx - i) ? p[2*id - i] : (mx - i));
- }
- else
- {
- p[i] = 1;
- }
当 mx - i > P[j] 的时候,以S[j]为中心的回文子串包含在以S[id]为中心的回文子串中,由于 i 和 j 对称,以S[i]为中心的回文子串必然包含在以S[id]为中心的回文子串中,所以必有 P[i] = P[j],见下图。
当 P[j] > mx - i 的时候,以S[j]为中心的回文子串不完全包含于以S[id]为中心的回文子串中,但是基于对称性可知,下图中两个绿框所包围的部分是相同的,也就是说以S[i]为中心的回文子串,其向右至少会扩张到mx的位置,也就是说 P[i] >= mx - i。至于mx之后的部分是否对称,就只能一个一个匹配了。
对于 mx <= i 的情况,无法对 P[i]做更多的假设,只能P[i] = 1,然后再去匹配了。
代码如下:
- #include <iostream>
- #include <string>
- #include <algorithm>
- using namespace std;
-
- void manacherPalindrome(const string &s){
- string str = "$#";
- for(int i = 0; i < s.size(); i++)
- {
- str += s[i];
- str += "#";
- }
- int *p = new int[str.size()];
- memset(p, 0, sizeof(int)*str.size());
-
- int mx = 0, id = 0;
-
- for(int i = 1; i < str.size(); i++){
- if(mx > i) p[i] = min(p[2*id - i], mx-i);
- else p[i] = 1;
-
- while(i+p[i] < str.size() && str[i-p[i]] == str[i+p[i]])
- p[i]++;
-
- if(i+p[i] > mx){
- mx = i + p[i];
- id = i;
- }
- }
-
-
- int maxSize = 0;
- int mid = 0;
- for(int i = 0; i < str.size(); i++){
- if(maxSize < p[i]){
- maxSize = p[i];
- mid = i;
- }
- }
- for(int k = mid - maxSize + 2; k < mid + maxSize; k=k+2){
- cout << str[k];
- }
-
- cout << endl;
-
- delete []p;
-
-
- }
-
- int main(){
- string s;
-
- while(cin >> s){
- manacherPalindrome(s);
- }
- return 0;
- }
此Manacher算法使用id、mx做配合,可以在每次循环中,直接对P[i]的快速赋值,从而在计算以i为中心的回文子串的过程中,不必每次都从1开始比较,减少了比较次数,最终使得求解最长回文子串的长度达到线性O(N)的时间复杂度。
// manacher算法,时间复杂度为O(N), 其中包含了4中拓扑结构,只有当i>=mx 或者p[2*id-i]=mx-i时需要扩展,而p[2*id-i]大于或者小于mx-i时不需要进行扩展
// 时间复杂度可以依据变量mx扩展的长度来计算,最多只能扩展到2N的位置(即加入#号后字符串的长度),故时间复杂度为O(N)
Hiho1032 最长回文子串
地址:http://hihocoder.com/problemset/problem/1032
代码:
- #include <memory.h>
- #include <iostream>
- #include <algorithm>
- #include <string>
- using namespace std;
-
- int findLongestPalindrome(const string &s){
- string str = "$#";
- for(int i = 0; i < s.size(); i++){
- str += s[i];
- str += "#";
- }
- int *p = new int[str.size()];
- memset(p, 0, sizeof(int)*str.size());
-
- int id = 0, mx = 0;
- for(int i = 1; i < str.size(); i++){
- if(mx > i) p[i] = min(p[2*id-i], mx-i);
- else p[i] = 1;
- while(i+p[i] < str.size() && str[i-p[i]]==str[i+p[i]])
- p[i]++;
- if(i+p[i] > mx){
- mx = i+ p[i];
- id = i;
- }
- }
- int maxSize = 0;
- for(int i = 0; i < str.size(); i++){
- if(maxSize < p[i])
- maxSize = p[i];
- }
- delete []p;
- return maxSize-1;
-
- }
-
- int main(){
- int N;
- cin >> N;
- while(N--){
- string s;
- cin >> s;
- cout << findLongestPalindrome(s) << endl;
- }
- return 0;
- }
-
-
-
-
-
-
-
-
-
-
-
-
-
参考文献:
1:http://www.cnblogs.com/en-heng/p/3973679.html最长回文子串
2:http://www.cnblogs.com/biyeymyhjob/archive/2012/10/04/2711527.htmlO(n)回文子串(Manacher)算法
3:https://github.com/julycoding/The-Art-Of-Programming-By-July/blob/master/ebook/zh/01.05.md
4:http://blog.csdn.net/kangroger/article/details/37742639
5:http://blog.163.com/zhaohai_1988/blog/static/2095100852012716105847112/