📜  编辑距离和 LCS(最长公共子序列)

📅  最后修改于: 2021-09-22 10:02:21             🧑  作者: Mango

在标准编辑距离中,我们允许进行 3 次操作,插入、删除和替换。考虑一个编辑距离的变体,我们只允许插入和删除两个操作,在这个变体中找到编辑距离。

例子:

Input : str1 = "cat", st2 = "cut"
Output : 2
We are allowed to insert and delete. We delete 'a'
from "cat" and insert "u" to make it "cut".

Input : str1 = "acb", st2 = "ab"
Output : 1
We can convert "acb" to "ab" by removing 'c'. 

一种解决方案是通过进行两次递归调用而不是三次调用来简单地修改“编辑距离解决方案”。一个有趣的解决方案是基于 LCS。
1) 找出两个字符串的LCS 。令 LCS 的长度为x
2) 设第一个字符串的长度为m ,第二个字符串的长度为n 。我们的结果是 (m – x) + (n – x)。我们基本上需要做(m-x)个删除操作和(n-x)个插入操作。

C++
// CPP program to find Edit Distance (when only two
// operations are allowed, insert and delete) using LCS.
#include
using namespace std;
 
int editDistanceWith2Ops(string &X, string &Y)
{
    // Find LCS
    int m = X.length(), n = Y.length();
    int L[m+1][n+1];
    for (int i=0; i<=m; i++)
    {
        for (int j=0; j<=n; j++)
        {
            if (i == 0 || j == 0)
                L[i][j] = 0;
            else if (X[i-1] == Y[j-1])
                L[i][j] = L[i-1][j-1] + 1;
            else
                L[i][j] = max(L[i-1][j], L[i][j-1]);
        }
    }   
    int lcs  = L[m][n];
 
    // Edit distance is delete operations +
    // insert operations.
    return (m - lcs) + (n - lcs);
}
 
/* Driver program to test above function */
int main()
{
    string X = "abc", Y = "acd";
    cout << editDistanceWith2Ops(X, Y);
    return 0;
}


Java
//Java program to find Edit Distance (when only two
// operations are allowed, insert and delete) using LCS.
 
class GFG {
 
    static int editDistanceWith2Ops(String X, String Y) {
        // Find LCS
        int m = X.length(), n = Y.length();
        int L[][] = new int[m + 1][n + 1];
        for (int i = 0; i <= m; i++) {
            for (int j = 0; j <= n; j++) {
                if (i == 0 || j == 0) {
                    L[i][j] = 0;
                } else if (X.charAt(i - 1) == Y.charAt(j - 1)) {
                    L[i][j] = L[i - 1][j - 1] + 1;
                } else {
                    L[i][j] = Math.max(L[i - 1][j], L[i][j - 1]);
                }
            }
        }
        int lcs = L[m][n];
 
        // Edit distance is delete operations +
        // insert operations.
        return (m - lcs) + (n - lcs);
    }
 
    /* Driver program to test above function */
    public static void main(String[] args) {
        String X = "abc", Y = "acd";
        System.out.println(editDistanceWith2Ops(X, Y));
 
    }
}
/* This Java code is contributed by 29AjayKumar*/


Python 3
# Python 3 program to find Edit Distance
# (when only two operations are allowed,
# insert and delete) using LCS.
 
def editDistanceWith2Ops(X, Y):
 
    # Find LCS
    m = len(X)
    n = len(Y)
    L = [[0 for x in range(n + 1)]
            for y in range(m + 1)]
    for i in range(m + 1):
        for j in range(n + 1):
            if (i == 0 or j == 0):
                L[i][j] = 0
            elif (X[i - 1] == Y[j - 1]):
                L[i][j] = L[i - 1][j - 1] + 1
            else:
                L[i][j] = max(L[i - 1][j],
                              L[i][j - 1])
 
    lcs = L[m][n]
 
    # Edit distance is delete operations +
    # insert operations.
    return (m - lcs) + (n - lcs)
 
# Driver Code
if __name__ == "__main__":
     
    X = "abc"
    Y = "acd"
    print(editDistanceWith2Ops(X, Y))
 
# This code is contributed by ita_c


C#
// C# program to find Edit Distance
// (when only two operations are
// allowed, insert and delete) using LCS.
using System;
 
class GFG
{
 
static int editDistanceWith2Ops(String X,
                                String Y)
{
    // Find LCS
    int m = X.Length, n = Y.Length;
    int [ , ]L = new int[m + 1 , n + 1];
    for (int i = 0; i <= m; i++)
    {
        for (int j = 0; j <= n; j++)
        {
            if (i == 0 || j == 0)
            {
                L[i , j] = 0;
            }
            else if (X[i - 1] == Y[j - 1])
            {
                L[i , j] = L[i - 1 , j - 1] + 1;
            }
            else
            {
                L[i , j] = Math.Max(L[i - 1 , j],
                                    L[i , j - 1]);
            }
        }
    }
    int lcs = L[m , n];
 
    // Edit distance is delete operations +
    // insert operations.
    return (m - lcs) + (n - lcs);
}
 
// Driver Code
public static void Main()
{
    String X = "abc", Y = "acd";
    Console.Write(editDistanceWith2Ops(X, Y));
}
}
 
// This code is contributed
// by 29AjayKumar


PHP


Javascript

//Javascript program to find Edit Distance (when only two
// operations are allowed, insert and delete) using LCS.
     
    function editDistanceWith2Ops(X,Y)
    {
        // Find LCS
        let m = X.length, n = Y.length;
        let L = new Array(m + 1);
        for(let i=0;i<L.length;i++)
        {
            L[i]=new Array(n+1);
            for(let j=0;j<L[i].length;j++)
            {
                L[i][j]=0;
            }
        }
        for (let i = 0; i <= m; i++) {
            for (let j = 0; j <= n; j++) {
                if (i == 0 || j == 0) {
                    L[i][j] = 0;
                } else if (X[i-1] == Y[j-1]) {
                    L[i][j] = L[i - 1][j - 1] + 1;
                } else {
                    L[i][j] = Math.max(L[i - 1][j], L[i][j - 1]);
                }
            }
        }
        let lcs = L[m][n];
  
        // Edit distance is delete operations +
        // insert operations.
        return (m - lcs) + (n - lcs);
    }
     
    /* Driver program to test above function */
    let X = "abc", Y = "acd";
    document.write(editDistanceWith2Ops(X, Y));
 
 
// This code is contributed by rag2127
2


输出:

2

时间复杂度: O(m * n)
辅助空间: O(m * n)

如果您希望与专家一起参加现场课程,请参阅DSA 现场工作专业课程学生竞争性编程现场课程