演算法:dp+字典樹
在CF刷字串題的時候遇到了這題,其實並沒有黑題這麼難,個人感覺最多是紫題吧(雖然一開始以為是字尾自動機的神仙題)。
首先注意到字串 \(s\) 長度很小( \(1\le|s|\le5000\) ),可以 \(\mathcal O(n^2)\) 地把所有子串求出來,再用Trie樹存起來,這樣就方便我們dfs求字典序第 \(k\) 小的半迴文串。所以問題重心變為怎麼快速判斷這些子串是否為半迴文串。根據半迴文串的特點,長度長的半迴文串是包含長度小的半迴文串的,所以我們可以用區間dp解決。設 \(f[i][l]\) 表示 \(s[i,i+l-1]\) 是否為半迴文串,它的轉移方程可以寫作( \([A]\) 表示 \(A\) 為真時值為1,否則為0):
\[\left\{\begin{array}{l}f[i][1]=1\\ f[i][2]=[s[i]=s[i+1]]\\f[i][3]=[s[i]=s[i+2]]\\f[i][4]=[s[i]=s[i+3]] \\ f[i][l]~=[~s[i]=s[i+l-1]\&\&f[i+2][l-4]~],l\ge5\end{array}\right.
\]
這樣dp的時間複雜度也是 \(\mathcal O(n^2)\) 的。之後dfs求解第 \(k\) 小的半迴文串就比較簡單了,由於Trie節點數也是 \(\mathcal O(n^2)\) 的,所以總時間複雜度是 \(\mathcal O(n^2)\) 的。
\(Code:\)
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 5e3 + 5;
bool f[N][N];
int trie[N * N][2],cnt,num[N * N],tot,k;
char s[N+6],ans[N];
void dfs(int now){
for(int i = 0;i <= 1;i++){
int next = trie[now][i];
if(next == 0) continue;
//如果該節點所表示的半迴文串數量大於k,說明答案就是該半迴文串。
if(k <= num[next]) {
ans[++tot] = i + 'a';
for(int j = 1; j <= tot; j++) printf("%c", ans[j]);
exit(0);
}
k -= num[next]; //k為全域性變數
ans[++tot] = i + 'a';
dfs(next);
tot--;
}
}
int main() {
scanf("%s %d",s+1,&k);
int len = strlen(s+1);
for(int i = 1;i <=len;i++) f[i][1] = 1;
for(int i = 1;i < len;i++) f[i][2] = s[i] == s[i+1];
for(int i = 1;i < len;i++) f[i][3] = s[i] == s[i+2];
for(int i = 1;i < len;i++) f[i][4] = s[i] == s[i+3];
for(int l = 5;l <= len;l++)
for(int i = 1; i <= len; i++)
f[i][l] = (s[i] == s[i+l-1] && f[i+2][l-4]);
//將s所有子串加入trie樹中
for(int i = 1;i <= len;i++){
int now = 0;
for(int l = 1; l+i-1 <= len; l++){
int c = s[i+l-1] - 'a';
if(trie[now][c] == 0) trie[now][c] = ++cnt;
now = trie[now][c];
if(f[i][l]) num[now]++;
}
}
dfs(0);
return 0;
}