hdu 3613 Best Reward (manachar算法)
Posted 风中的簌雨
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了hdu 3613 Best Reward (manachar算法)相关的知识,希望对你有一定的参考价值。
Best Reward
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)
Problem Description
After an uphill battle, General Li won a great victory. Now the head of state decide to reward him with honor and treasures for his great exploit.
One of these treasures is a necklace made up of 26 different kinds of gemstones, and the length of the necklace is n. (That is to say: n gemstones are stringed together to constitute this necklace, and each of these gemstones belongs to only one of the 26 kinds.)
In accordance with the classical view, a necklace is valuable if and only if it is a palindrome - the necklace looks the same in either direction. However, the necklace we mentioned above may not a palindrome at the beginning. So the head of state decide to cut the necklace into two part, and then give both of them to General Li.
All gemstones of the same kind has the same value (may be positive or negative because of their quality - some kinds are beautiful while some others may looks just like normal stones). A necklace that is palindrom has value equal to the sum of its gemstones‘ value. while a necklace that is not palindrom has value zero.
Now the problem is: how to cut the given necklace so that the sum of the two necklaces‘s value is greatest. Output this value.
One of these treasures is a necklace made up of 26 different kinds of gemstones, and the length of the necklace is n. (That is to say: n gemstones are stringed together to constitute this necklace, and each of these gemstones belongs to only one of the 26 kinds.)
In accordance with the classical view, a necklace is valuable if and only if it is a palindrome - the necklace looks the same in either direction. However, the necklace we mentioned above may not a palindrome at the beginning. So the head of state decide to cut the necklace into two part, and then give both of them to General Li.
All gemstones of the same kind has the same value (may be positive or negative because of their quality - some kinds are beautiful while some others may looks just like normal stones). A necklace that is palindrom has value equal to the sum of its gemstones‘ value. while a necklace that is not palindrom has value zero.
Now the problem is: how to cut the given necklace so that the sum of the two necklaces‘s value is greatest. Output this value.
Input
The first line of input is a single integer T (1 ≤ T ≤ 10) - the number of test cases. The description of these test cases follows.
For each test case, the first line is 26 integers: v1, v2, ..., v26 (-100 ≤ vi ≤ 100, 1 ≤ i ≤ 26), represent the value of gemstones of each kind.
The second line of each test case is a string made up of charactor ‘a‘ to ‘z‘. representing the necklace. Different charactor representing different kinds of gemstones, and the value of ‘a‘ is v1, the value of ‘b‘ is v2, ..., and so on. The length of the string is no more than 500000.
For each test case, the first line is 26 integers: v1, v2, ..., v26 (-100 ≤ vi ≤ 100, 1 ≤ i ≤ 26), represent the value of gemstones of each kind.
The second line of each test case is a string made up of charactor ‘a‘ to ‘z‘. representing the necklace. Different charactor representing different kinds of gemstones, and the value of ‘a‘ is v1, the value of ‘b‘ is v2, ..., and so on. The length of the string is no more than 500000.
Output
Output a single Integer: the maximum value General Li can get from the necklace.
Sample Input
2
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1
aba
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1
acacac
Sample Output
1
6
题目大意:
有一个由26种宝石组成的项链,给定每一种宝石的的价值,现需要将这串项链分成两部分
如果分开的项链是一个回文的则它的价值为之一部分项链上所有宝石价值的总和,否则的话它的价值为0
求能够得到的最大价值是多少。
解题思路:
先算出这串项链的价值前缀和,然后用manachar算法求出项链的p数组,根据p数组枚举切割点,从而求得最大价值。
AC代码:
1 #include <stdio.h> 2 #include <string.h> 3 #include <algorithm> 4 using namespace std; 5 6 const int N = 500010; 7 char s[N*2]; // 项链 8 int val[26], p[N*2], sum[N]; // val存每种宝石的价值 sum存项链的价值前缀和(前i个字符价值和) 9 int pre[N], pos[N]; // pre标记前i个字符为回文串 pos标记后i个字符为回文串 10 int manachar () 11 { 12 memset(pos,0,sizeof(pos)); 13 memset(pre,0,sizeof(pre)); 14 memset(sum,0,sizeof(sum)); 15 int i, j = 0; 16 int len = strlen(s); 17 for (i = 1; i <= len; i ++) // 求项链价值的前缀和 18 sum[i] += sum[i-1]+val[s[i-1]-‘a‘]; 19 20 for (i = len; i >= 0; i --) // 预处理项链字符串 21 { 22 s[i*2+2] = s[i]; 23 s[i*2+1] = ‘#‘; 24 }s[0] = ‘@‘; 25 // puts(s); 26 27 int id = 0, mx = 0; 28 for (i = 2; s[i]; i ++) 29 { 30 p[i] = i<mx? min(p[id*2-i],mx-i) : 1; 31 while (s[i+p[i]] == s[i-p[i]]) p[i] ++; 32 if (i+p[i] > mx){ 33 id = i; 34 mx = i + p[i]; 35 } 36 if (i == p[i]) pre[p[i]-1] = 1; //表示前缀(前p[i]-1个字符)是回文串 37 if (i+p[i] == 2*len+2) pos[p[i]-1] = 1; //表示后缀(后p[i]-1个字符)是回文串 38 } 39 int ans = -0x3f3f3f3f; 40 for (i = 1; i < len; i ++) // 不断枚举切割点 求最大价值 41 { 42 j = 0; 43 if (pre[i]) j += sum[i]; 44 if (pos[len-i]) j += sum[len]-sum[i]; 45 if (j > ans) ans = j; 46 } 47 return ans; 48 } 49 int main () 50 { 51 int t,i; 52 scanf("%d",&t); 53 while (t --) 54 { 55 for (i = 0; i < 26; i ++) 56 scanf("%d",&val[i]); 57 scanf("%s",s); 58 int ans = manachar(); 59 printf("%d\n",ans); 60 } 61 return 0; 62 }
以上是关于hdu 3613 Best Reward (manachar算法)的主要内容,如果未能解决你的问题,请参考以下文章
HDU 3613 Best Reward(拓展KMP算法求解)
hdu 3613 Best Reward (manachar算法)