字符串末尾添加字符构造回文
对于一个字符串,我们想通过添加字符的方式使得新的字符串整体变成回文串,但是只能在原串的结尾添加字符,请返回在结尾添加的最短字符串。
给定原字符串A及它的长度n,请返回添加的字符串。保证原串不是回文串。
Manachert算法告诉我们一件事:任何一个位置为中心,向左右两边扩的回文半径或回文直径的范围都能获取到
这道题其实是在求这样一件事:必须包含最后一个字符的情况下,哪一段字符的回文串是最长的? (第一个能包含最后一个字符串的回文直径)排除,把前面的逆序就是答案
public class Palindrome { public static char[] manacherString(String str) { char[] charArr = str.toCharArray(); char[] res = new char[str.length() * 2 + 1]; int index = 0; for (int i = 0; i != res.length; i++) { res[i] = (i & 1) == 0 ? '#' : charArr[index++]; } return res; } public String addToPalindrome(String str, int n) { // write code here if (str == null || str.length() == 0) { return null; } char[] charArr = manacherString(str); int[] pArr = new int[charArr.length]; int index = -1; int pR = -1; int maxContainsEnd = -1; for (int i = 0; i != charArr.length; i++) { pArr[i] = pR > i ? Math.min(pArr[2 * index - i], pR - i) : 1; while (i + pArr[i] < charArr.length && i - pArr[i] > -1) { if (charArr[i + pArr[i]] == charArr[i - pArr[i]]) pArr[i]++; else { break; } } if (i + pArr[i] > pR) { pR = i + pArr[i]; index = i; } if (pR == charArr.length) { maxContainsEnd = pArr[i]; break; } } char[] res = new char[str.length() - maxContainsEnd + 1]; for (int i = 0; i < res.length; i++) { res[res.length - 1 - i] = charArr[i * 2 + 1]; } return String.valueOf(res); } }