Hdu 4513 吉哥系列故事——完美队形II (manacher变形)

时间:2022-04-28 00:27:31

题目链接:

  http://acm.hdu.edu.cn/showproblem.php?pid=4513

题目描述:

  打完题目描述了,点开题目,发现题目是中文,orz.jpg。果断又删掉了,习惯真可怕.......

解题思路:

  刚开始,我先manacher求出以 i 为中心的回文串半径存入vis,然后暴力循环每一个位置是不是最长的完美队列。果断T了,胡乱改了几处依旧T。突然灵机一动,发现可以在跑vis数组的时候加一些判定条件,然后直接求出max(vis[i])即可。

  学习算法是次要,理解算法思想,锻炼思维灵敏逻辑缜密才最重要。

 #include <cstdio>
#include <string>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <cmath>
#include <queue>
using namespace std; typedef long long LL;
const int maxn = ;
int vis[maxn*], a[maxn], b[maxn*]; bool judge (int x, int y)
{
for (int i=x+; i<=y/; i++)
if (a[i]<a[i-])
return false;
return true;
} void manacher (int s[], int len)
{
int l = ;
b[l ++] = -;
b[l ++] = -; for (int i=; i<len; i++)
{
b[l ++] = a[i];
b[l ++] = -;
}
b[l] = ; int mx = , id = , ans = ; for (int i=; i<l; i++)
{
vis[i] = mx > i ? min (vis[*id-i], mx-i):; while (b[i-vis[i]] == b[i+vis[i]] && (b[i-vis[i]]==- || b[i-vis[i]]<=b[i-vis[i]+])) vis[i] ++; if (i + vis[i] > mx)
{
mx = i + vis[i];
id = i;
}
ans = max (ans, vis[i]);
} printf ("%d\n", ans - );
} int main ()
{
int T, n;
scanf ("%d", &T); while (T --)
{
scanf ("%d", &n);
for (int i=; i<n; i++)
scanf ("%d", &a[i]); manacher(a, n);
}
return ;
}