CF 1562 E. Rescue Niwen! (字符串+DP+后缀数组优化)

12 篇文章 0 订阅

E. Rescue Niwen!

题意:

给定一个字符串 s s s,将 s s s 的所有子串按照它在 s s s 中的出现位置 l , r l,r l,r排成一列,其中 l l l 为第一关键字 r r r 为第二关键字。
求这个字符串序列的最长上升子序列,其中大于的定义是字典序大于。

分析:

首先我们从题意中看,他要找最长上升的子序列,那么首先我们需要知道,第 x x x位置开头的子序列集合我们用 S ( x ) S(x) S(x)表示,那么如果我们取了第 x x x位置开的的子序列那么我们取的最长上升子序列,一定是在 S ( x ) S(x) S(x)中连续的。

原因:
x x x位置元素大于前面的最长上升子序列时,那么从 x ( x + 1 ) x(x+1) x(x+1)一定> x x x这个,然后递归下去 如果后面有位置与 x x x位置元素一样,那么他后面部分位置可能会取代或者填充 S ( x ) S(x) S(x)位置。

所以我们要找到这个取代或者填充的位置,那么我们就需要维护一个字符串一致的长度,直接转移过去询问。这个就是后缀数组优化。
ORZ

int dp[5100][5100];
int d[5100];
int n;
string str;
void solve()
{
    cin >> n >> str;
    str = " " + str;
    dp[n + 1][n + 1] = dp[n + 1][n] = dp[n][n + 1] = 0;
    for(int i = n; i >= 1; i--)
    {
        for(int j = n; j >= 1; j--)
        {
            if(str[i] == str[j]) dp[i][j] = dp[i + 1][j + 1] + 1; //减少复杂度,直接找到不同的地方
            else dp[i][j] = 0;
        }
    }

    int ans = 0;
    for(int i = 1; i <= n; i++)
    {
        d[i] = n - i + 1; ///全选
        for(int j = 1; j < i; j++) ///看前面都可以选那些
        {
            int len = dp[i][j]; ///找到不同的地方
            if(i + len - 1 >= n || str[i + len] <= str[j + len]) continue;
            d[i] = max(d[i], d[j] + n - i + 1 - len);
        }
        ans = max(ans, d[i]);
    }
    printf("%d\n", ans);
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值