Anonymous Goose Blog
Published on

A Bonus Classic DP

Common Child

source

Oh, the classic LCS.

Let SS and TT be the first and second strings and SiS_i represent the ii-th character in stringSS, we can define our recursive solution as follows:

  • ifSj=Tilcs(i,j)=lcs(i1,j1)+1if S_j = T_i \Rightarrow lcs(i, j) = lcs(i - 1, j - 1) + 1
  • ifSjTilcs(i,j)=max(lcs(i1,j),lcs(i,j1))if S_j \neq T_i \Rightarrow lcs(i, j) = \max(lcs(i - 1, j), lcs(i, j - 1))

In other words, if we look at the last characters of both strings we have only three possibilities:

  • The characters match, so we can add 1 to our answer and remove them both.
  • We remove the last character from SSand look for the LCS of SS' and TT.
  • We remove the last character from TTand look for the LCS of SS and TT'.

The rest is simply a matter of implementing the bottom-up solution.

Note: I tried submitting a memoized recursive solution afterwards but it TLE'd in a couple of cases
#include<bits/stdc++.h>

using namespace std;

#define N 5010
#define max(x,y) x >= y ? x : y

int dp[N][N];

int main(){
    ios::sync_with_stdio(false);
    string s,t;
    cin >> s >> t;
    int n = (int)s.length();
    for(int i = 1; i <= n; i++){
        for(int j = 1; j <= n; j++){
            if(s[i-1] == t[j-1])
                dp[i][j] = dp[i-1][j-1] + 1;
            else dp[i][j] = max(dp[i][j-1], dp[i-1][j]);
        }
    }
    cout << dp[n][n] << endl;
    return 0;
}