码迷,mamicode.com
首页 > 其他好文 > 详细

hdu 5371 Hotaru's problem【manacher】

时间:2015-08-13 06:32:08      阅读:121      评论:0      收藏:0      [点我收藏+]

标签:

题目链接:
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

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!