【DP && 两个字符串求 k 个顺序子串相同最大长度】CodeForces - 682D Alyona and Strings

Step1 Problem

给你两个长度分别为n, m的字符串,在第一个字符串找 k 个顺序的子串,在第二个字符串中均出现其顺序一样,求最大这些子串长度和

Step2 Ideas:

dp[i][j][k][1]:第一个串第 i 位, 第二个串第 j 位,当中 k 个连续的子串,s1[i] == s2[j] 且是最后一个子串末尾
dp[i][j][k][0]:第一个串第 i 位, 第二个串第 j 位,当中 k 个连续的子串,s1[i] 和 s2[j] 不同时是最后一个子串末尾
由于 dp[i][j][k][0] 最后一个子串末尾不同时是 s1[i] 和 s2[j] :
dp[i][j][k][0] = max(dp[i][j][k][0], dp[i-1][j][k][0]);
dp[i][j][k][0] = max(dp[i][j][k][0], dp[i][j-1][k][0]);
dp[i][j][k][0] = max(dp[i][j][k][0], dp[i-1][j][k][1]);
dp[i][j][k][0] = max(dp[i][j][k][0], dp[i][j-1][k][1]);
当 s1[i] == s2[j] 时 :两种情况,加入到上一个子串,自己单独作为一个子串。
dp[i][j][k][1] = max(dp[i-1][j-1][k][1], dp[i-1][j-1][k-1][0])+1;

Step3 Code:

#include<bits/stdc++.h>
using namespace std;
const int N = 1005;
int dp[N][N][15][2];
char s1[N], s2[N];
int main()
{
    int n, m, K;
    scanf("%d %d %d", &n, &m, &K);
    scanf("%s %s", s1+1, s2+1);
    for(int i = 1; i <= n; i++)
    {
        for(int j = 1; j <= m; j++)
        {
            if(s1[i] == s2[j]) {
                for(int k = 1; k <= K; k++)
                {
                    dp[i][j][k][1] = max(dp[i-1][j-1][k][1], dp[i-1][j-1][k-1][0])+1;
                }
            }
            for(int k = 1; k <= K; k++)
            {
                dp[i][j][k][0] = max(dp[i][j][k][0], dp[i-1][j][k][0]);
                dp[i][j][k][0] = max(dp[i][j][k][0], dp[i][j-1][k][0]);
                dp[i][j][k][0] = max(dp[i][j][k][0], dp[i-1][j][k][1]);
                dp[i][j][k][0] = max(dp[i][j][k][0], dp[i][j-1][k][1]);
            }
        }
    }
    printf("%d\n", max(dp[n][m][K][0], dp[n][m][K][1]));
}
展开阅读全文

没有更多推荐了,返回首页