# 583. Delete Operation for Two Strings - LeetCode Best Practices Visit original link: [583. Delete Operation for Two Strings - LeetCode Best Practices](https://leetcoder.net/en/leetcode/583-delete-operation-for-two-strings) for a better experience! LeetCode link: [583. Delete Operation for Two Strings](https://leetcode.com/problems/delete-operation-for-two-strings), difficulty: **Medium**. ## LeetCode description of "583. Delete Operation for Two Strings" Given two strings `word1` and `word2`, return the **minimum** number of **steps** required to make `word1` and `word2` the same. In one **step**, you can delete exactly one character in either string. ### [Example 1] **Input**: `word1 = "sea", word2 = "eat"` **Output**: `2` **Explanation**:
You need one step to make "sea" to "ea" and another step to make "eat" to "ea".
### [Example 2] **Input**: `word1 = "leetcode", word2 = "etco"` **Output**: `4` ### [Constraints] - `1 <= word1.length, word2.length <= 500` - `word1` and `word2` consist of only lowercase English letters. ## Intuition It is a question of **comparing two strings**. After doing similar questions many times, we will develop an intuition to use `dynamic programming with two-dimensional arrays`. ## Steps ### Common steps in dynamic programming These five steps are a pattern for solving `dynamic programming` problems. 1. Determine the **meaning** of the `dp[i][j]` - Since there are two strings, we can use two-dimensional arrays as the default option. - At first, try to use the problem's `return` value as the value of `dp[i][j]` to determine the meaning of `dp[i][j]`. If it doesn't work, try another way. - `dp[i][j]` represents the **minimum** number of steps required to make `word1`'s first `i` letters and `word2`'s first `j` letters the same. - `dp[i][j]` is an integer. 2. Determine the `dp` array's initial value - Use an example: ``` After initialization, the 'dp' array would be: # e a t # 0 1 2 3 # dp[0] # s 1 0 0 0 # e 2 0 0 0 # a 3 0 0 0 ``` - `dp[0][j] = j`, because `dp[0]` represents the empty string, and the number of steps is just the number of chars to be deleted. - `dp[i][0] = i`, the reason is the same as the previous line, just viewed in vertical direction. 3. Determine the `dp` array's recurrence formula - Try to complete the grid. In the process, you will get inspiration to derive the formula. ``` 1. word1 = "s", word2 = "eat" # e a t # 0 1 2 3 # s 1 2 3 4 # dp[1] ``` ``` 2. word1 = "se", word2 = "eat" # e a t # 0 1 2 3 # s 1 2 3 4 # e 2 1 2 3 ``` ``` 3. word1 = "sea", word2 = "eat" # e a t # 0 1 2 3 # s 1 2 3 4 # e 2 1 2 3 # a 3 2 1 2 ``` - When analyzing the sample `dp` grid, remember there are three important points which you should pay special attention to: `dp[i - 1][j - 1]`, `dp[i - 1][j]` and `dp[i][j - 1]`. The current `dp[i][j]` often depends on them. - If the question is also true in reverse (swap `word1` and `word2`), and we need to use `dp[i - 1][j]` or `dp[i][j - 1]`, then we probably need to use both of them. - We can derive the `Recurrence Formula`: ```python if word1[i - 1] == word2[j - 1] dp[i][j] = dp[i - 1][j - 1] else dp[i][j] = min(dp[i - 1][j], dp[i][j - 1]) + 1 ``` 4. Determine the `dp` array's traversal order - `dp[i][j]` depends on `dp[i - 1][j - 1]`, `dp[i - 1][j]` and `dp[i][j - 1]`, so we should traverse the `dp` array from top to bottom, then from left to right. 5. Check the `dp` array's value - Print the `dp` to see if it is as expected. ## Complexity - Time complexity: `O(N * M)`. - Space complexity: `O(N * M)`. ## Java ```java class Solution { public int minDistance(String word1, String word2) { var dp = new int[word1.length() + 1][word2.length() + 1]; for (var i = 0; i < dp.length; i++) { dp[i][0] = i; } for (var j = 0; j < dp[0].length; j++) { dp[0][j] = j; } for (var i = 1; i < dp.length; i++) { for (var j = 1; j < dp[0].length; j++) { if (word1.charAt(i - 1) == word2.charAt(j - 1)) { dp[i][j] = dp[i - 1][j - 1]; } else { dp[i][j] = Math.min(dp[i - 1][j], dp[i][j - 1]) + 1; } } } return dp[dp.length - 1][dp[0].length - 1]; } } ``` ## C# ```csharp public class Solution { public int MinDistance(string word1, string word2) { var dp = new int[word1.Length + 1, word2.Length + 1]; for (var i = 0; i < dp.GetLength(0); i++) dp[i, 0] = i; for (var j = 0; j < dp.GetLength(1); j++) dp[0, j] = j; for (var i = 1; i < dp.GetLength(0); i++) { for (var j = 1; j < dp.GetLength(1); j++) { if (word1[i - 1] == word2[j - 1]) { dp[i, j] = dp[i - 1, j - 1]; } else { dp[i, j] = Math.Min(dp[i - 1, j], dp[i, j - 1]) + 1; } } } return dp[dp.GetUpperBound(0), dp.GetUpperBound(1)]; } } ``` ## Python ```python class Solution: def minDistance(self, word1: str, word2: str) -> int: dp = [[0] * (len(word2) + 1) for _ in range(len(word1) + 1)] for i in range(len(dp)): dp[i][0] = i for j in range(len(dp[0])): dp[0][j] = j for i in range(1, len(dp)): for j in range(1, len(dp[0])): if word1[i - 1] == word2[j - 1]: dp[i][j] = dp[i - 1][j - 1] else: dp[i][j] = min(dp[i - 1][j], dp[i][j - 1]) + 1 return dp[-1][-1] ``` ## C++ ```cpp class Solution { public: int minDistance(string word1, string word2) { vector