一、题目描述
二、解题思路
该题目使用动态规划的思想来解决问题
刚开始我还在想,删除+添加的操作可以等价为替换操作,如果替换操作的Cost大于删除+添加组合操作的Cost之和就需要把 rc=dc+ic。
但是在动态规划中,如果对三种不同的操作方式进行比较然后取较小值,不用进行上面的替换操作就可以达到效果,假设i表示指向str2的指针,j表示指向str1的指针,将str1->str2的过程中:
无非就是一个从对角线 [i-1][j-1] ->(替换) [i][j] cost=rc
另一个从 [i-1][j-1] ->(删除) [i-1][j] ->(添加) [i][j] cost=dc+ic
所以这里的状态转移方程就是:
deleteCost = dp[i][j-1]+dc;
addCost = dp[i-1][j]+ic;
replaceCost = dp[i-1][j-1]+rc;
dp[i][j] = Min(deleteCost,addCost,replaceCost)
初始化dp数组:
dp[0][j] = j*dc :代表str2为空串,str1只能删除,则花费为 j × dc
dp[i][0] = i*ic :代表str1为空串,str1只能添加,则花费为 i × ic
三、代码实现
import java.util.*;public class Solution {/*** 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可** min edit cost* @param str1 string字符串 the string* @param str2 string字符串 the string* @param ic int整型 insert cost* @param dc int整型 delete cost* @param rc int整型 replace cost* @return int整型*/public int minEditCost (String str1, String str2, int ic, int dc, int rc) {//使用动态规划找两个字符串的最小代价int[][] maxCostArr=new int[str2.length()+1][str1.length()+1];//dp数组进行初始化//初始化第一行,当str2为空字符串时for(int i=0;i<=str1.length();i++){maxCostArr[0][i]=i*dc;}//初始化第一列,当str1为空字符串时for(int i=0;i<=str2.length();i++){maxCostArr[i][0]=i*ic;}//双层循环对dp数组进行修改for(int p2=1;p2<=str2.length();p2++){for(int p1=1;p1<=str1.length();p1++){if(str2.charAt(p2-1)==str1.charAt(p1-1)){//Cost不需要增加maxCostArr[p2][p1]=maxCostArr[p2-1][p1-1];}else{//计算三种情况下的代价值,去最小值int deleteCost=maxCostArr[p2][p1-1]+dc;int addCost=maxCostArr[p2-1][p1]+ic;int replaceCost=maxCostArr[p2-1][p1-1]+rc;maxCostArr[p2][p1]=Math.min(deleteCost,Math.min(addCost,replaceCost));}}}return maxCostArr[str2.length()][str1.length()];}
}
四、刷题链接
编辑距离(二)_牛客题霸_牛客网