题意:给定一个字符串s和一个整数k,可以将字符串的某个字母更改为任意字母,求将s分割为k段,每段都为回文串的最少改动次数。
题解:dp[i][j]表示从i-j的字符串变为回文串的最少修改次数。然后记忆化搜索即可。
AC代码:
class Solution {
public:
map, int> Q;
int dp[107][107];
int n;
int palindromePartition(string s, int k) {
Q.clear();
n = s.length();
init(s);
return dfs(s, 0, k);
}
void init(string s) {
for (int i = 0; i < n - 1; i++) {
dp[i][i] = 0;
dp[i][i + 1] = s[i] == s[i + 1] ? 0 : 1;
}
for (int i = 2; i < n; i++) {
for (int j = i - 2; j >= 0; j--) {
dp[j][i] = dp[j + 1][i - 1] + (s[i] != s[j] ? 1 : 0);
}
}
}
int dfs(string s, int i, int k) {
if (Q.find(make_pair(i, k)) != Q.end()) return Q[make_pair(i, k)];
if (k == 1) return dp[i][n - 1];
int res = n + 1;
for (int j = i + 1; j < n; j++) {
res = min(res, dp[i][j - 1] + dfs(s, j, k - 1));
}
Q[make_pair(i, k)] = res;
return res;
}
};