Maior Subsequência Comum

Conhecido como Longest Common Subsequence, este é um algoritmo para obter o tamanho da maior subsequência comum entre duas strings.

Motivação

Dadas duas strings A e B, a maior subsequência comum se dá pela maior sequência de letras possível que está tanto em A quanto em B na ordem correta. Isso não significa que todas as letras precisam ser consecutivas nas duas strings, mas é importante que a ordem seja exatamente a mesma nas duas strings. Por exemplo, se A = "ABCD" e B = "ACBAD", então a maior subsequência comum entre essas duas string é "ABD" ou "ACD" (ambas tamanho 3). Perceba que apesar das letras não estarem consecutivas, a ordem foi respeitada. Outros exemplos estão esquematizados na tabela abaixo:

String A

String B

Maior Subsequência Comum

"ABCDGH"

"AEDFHR"

"ADH" (tamanho 3)

"AGGTAB"

"GXTXAYB"

"GTAB" (tamanho 4)

"Hey This java is hot"

"Java is a new paradigm"

"ava is " (tamanho 7)

Tal algoritmo é amplamente usado quando queremos identificar diferenças entre dois textos, o que é a base do comando diff e do controle de versões de ferramentas como o git, também utilizado em computação biomédica, na comparação de sequência de genomas.

Algoritmos

Maior Subsequência Comum

A partir daqui, vamos considerar que estamos interessados no algoritmo que retorna apenas o tamanho da maior subsequência comum entre duas strings. Vamos chamar tal função que realiza essa tarefa de LCS(A, B), onde A e B são as duas strings que se deseja saber qual é o tamanho da maior subsequência comum entre elas. Também vamos considerar que |A| = n e |B| = m, ou seja, os tamanhos das strings A e B são respectivamente n e m.

Este problema pode ser resolvido eficientemente com complexidade O(nm) usando programação dinâmica, já que o problema de determinar LCS(A, B) pode ser dividido em problemas menores e que podemos ir resolvendo passo a passo usando resultados anteriores para nos guiar nos problema maiores. Isso pode ser feito porque a nossa função LCS(A, B) tem duas propriedades muito interessantes:

  • Se adicionarmos a mesma letra X ao final de A e B, temos que LCS(A + X, B + X) = LCS(A, B) + 1, pois adicionando a mesma letra, temos certeza que estamos agregando à maior subsequência comum, sem perda de ordenação, já que Y é a última letra de ambas as strings.

  • Se adicionarmos letras diferentes ao final de A e B, não podemos adicionar 1 tão fácil, já que sabemos que as letras diferentes não podem ambas fazer parte da mesma subsequência. Mas podemos dizer que LCS(A + X, B + Y) = max(LCS(A + X, B), LCS(A, B + Y)), já que podemos basear nossa resposta numa simples questão de "sabemos que as duas letras juntas não vão formar uma subsequência maior, mas uma delas sozinha pode contribuir para uma subsequência maior".

Com isso, temos a seguinte recorrência:

LCS(A,B)={0,se A e B sa˜o strings vaziasLCS(A,B)+1,se a uˊltima letra de A e B sa˜o iguaismax(LCS(A,B),LCS(A,B)),se a uˊltima letra de A e B na˜o sa˜o iguaisonde A’ e B’ sa˜o equivalente aˋs strings A e B sem suas respectivas uˊltimas letrasLCS(A, B) = \begin{cases} 0, \text{se } A \text{ e } B \text{ são strings vazias}\\ LCS(A', B') + 1, \text{se a última letra de A e B são iguais}\\ max(LCS(A, B'), LCS(A', B)), \text{se a última letra de A e B não são iguais} \end{cases}\\ \text{onde A' e B' são equivalente às strings A e B sem suas respectivas últimas letras}

E, com isso, temos o código

int LCS(char* A, char* B){
    int** T;
    int n, m;
    
    n = strlen(A), m = strlen(B);
    
    T = (int**) malloc(sizeof(int*) * (n + 1));
    for(int i = 0; i <= n; ++i){
        T[i] = (int*) malloc(sizeof(int) * (m + 1));
        for(int j = 0; j <= m; ++j){
            T[i][j] = 0;
        }
    }
    
    for(int i = 1; i <= n; ++i){
        for(int j = 1; j <= m; ++j){
            if(A[i - 1] == B[j - 1]){
                T[i][j] = T[i - 1][j - 1] + 1;
            }else{
                T[i][j] = T[i][j - 1];
                T[i][j] = T[i][j] < T[i - 1][j] ? T[i - 1][j] : T[i][j];
            }
        }
    }
    
    return T[n][m];
}

Maior Substring Comum

Se estamos considerando apenas substrings, onde além da ordem, as letras agora também precisam ser consecutivas, então a segunda propriedade demonstrada no algoritmo anterior não é mais aplicada aqui, ou seja, se uma string termina com letras diferentes, não tem maneira alguma de uma das letras contribuir para a solução. Mas a primeira propriedade, por outro lado, segue valendo, já que a última letra sendo igual significa que ela pode contribuir para a resposta para o caso anterior sem as letras.

Logo, precisamos aplicar apenas uma pequena mudança em relação ao algoritmo anterior:

int LCS(char* A, char* B){
    int** T;
    int n, m, resultado;
    
    n = strlen(A), m = strlen(B);
    
    T = (int**) malloc(sizeof(int*) * (n + 1));
    for(int i = 0; i <= n; ++i){
        T[i] = (int*) malloc(sizeof(int) * (m + 1));
        for(int j = 0; j <= m; ++j){
            T[i][j] = 0;
        }
    }
    
    resultado = 0;
    for(int i = 1; i <= n; ++i){
        for(int j = 1; j <= m; ++j){
            if(A[i - 1] == B[j - 1]){
                T[i][j] = T[i - 1][j - 1] + 1;
                resultado = resultado < T[i][j] ? T[i][j] : resultado;
            }
        }
    }
    
    return resultado;
}

Onde é necessário definir um máximo global e ir analisando casa a casa porque não é possível acumular resultados da uma posição da matriz para a outra, pois isso poderia atrapalhar os cálculos realizados.

Problemas

Last updated