Ann and Half-Palindrome题解

算法:dp+字典树

在博客园观看本文

题目链接Ann and Half-Palindrome

  在CF刷字符串题的时候遇到了这题,其实并没有黑题这么难,个人感觉最多是紫题吧(虽然一开始以为是后缀自动机的神仙题)。

  首先注意到字符串 s s s 长度很小( 1 ≤ ∣ s ∣ ≤ 5000 1\le|s|\le5000 1s5000 ),可以 O ( n 2 ) \mathcal O(n^2) O(n2) 地把所有子串求出来,再用Trie树存起来,这样就方便我们dfs求字典序第 k k k 小的半回文串。所以问题重心变为怎么快速判断这些子串是否为半回文串。根据半回文串的特点,长度长的半回文串是包含长度小的半回文串的,所以我们可以用区间dp解决。设 f [ i ] [ l ] f[i][l] f[i][l] 表示 s [ i , i + l − 1 ] s[i,i+l-1] s[i,i+l1] 是否为半回文串,它的转移方程可以写作( [ A ] [A] [A] 表示 A A A 为真时值为1,否则为0):
{ 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 ≥ 5 \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. f[i][1]=1f[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+l1]&&f[i+2][l4] ],l5
  这样dp的时间复杂度也是 O ( n 2 ) \mathcal O(n^2) O(n2) 的。之后dfs求解第 k k k 小的半回文串就比较简单了,由于Trie节点数也是 O ( n 2 ) \mathcal O(n^2) O(n2) 的,所以总时间复杂度是 O ( n 2 ) \mathcal O(n^2) O(n2) 的。

C o d e : Code: 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;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ailanxier

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值