标签:
题目链接:
http://acm.hdu.edu.cn/showproblem.php?pid=5371
题意:
给出一个长度为n的串,要求找出一条最长连续子串。这个子串要满足:1:可以平均分成三段,2:第一段和第三段相等,3:第一段和第二段回文。求最大子串的长度。
代码:
#include<stdio.h>
#include<iostream>
#include<math.h>
#include<stdlib.h>
#include<ctype.h>
#include<algorithm>
#include<vector>
#include<string>
#include<queue>
#include<stack>
#include<set>
#include<map>
#include<string.h>
using namespace std;
const int N = 1100550;
int n, l, tmp;
int p[2 * N];//记录回文半径
int str0[N];//原始串
int str[2 * N];//转换后的串
void init()
{
int i;
str[0] = -2; str[1] = -1;
l = 2;
for (i = 0, l = 2; i<n; i++, l += 2)
{
str[l] = str0[i];
str[l + 1] = -1;
}
str[l] = -5;
}
int solve()
{
int i, mx, id;
mx = 0;//mx即为当前计算回文串最右边字符的最大值
for (i = 1; i < l; i++)
{
if (mx>i)
p[i] = p[2 * id - i]>(mx - i) ? (mx - i) : p[2 * id - i];
else
p[i] = 1;//如果i>=mx,要从头开始匹配
while (str[i + p[i]] == str[i - p[i]])
p[i]++;
if (i + p[i]>mx)//若新计算的回文串右端点位置大于mx,要更新po和mx的值
{
mx = i + p[i];
id = i;
}
}
int ans = 0;
for (int i = 1; i < l; i += 2)
for (int j = i + p[i] - 1; j - i > ans; j -= 2)
{
if (p[j] >= j-i+1 && ans < j - i)
{
ans = j - i;
break;
}
}
return ans / 2 * 3;
}
int main()
{
int t;
int cases = 1;
scanf("%d", &t);
while (t--)
{
scanf("%d", &n);
for (int i = 0; i < n; i++)
scanf("%d",&str0[i]);
init();
printf("Case #%d: %d\n", cases++, solve());
}
return 0;
}
版权声明:转载请注明出处。
hdu 5371 Hotaru's problem【manacher】
标签:
原文地址:http://blog.csdn.net/u014427196/article/details/47487445