问题描述

输入 原字符串StrOrg,目标字符串StrTarget,插入、删除、替换的编辑代价ic,dc,rc。输出将原字符串编辑成目标字符串的最小代价。

解题思路

状态表示

dp[i][j]表示把strOrg[0:i]编辑成strTarget[0:j]的最小代价。

状态转移方程

从以下三种状态的取最小即可:

  • 删除:dp[i][j]=dp[i-1][j]+dc。原串删除最后一个元素,再转移到目标串
  • 插入:dp[i][j]=dp[i][j-1]+ic。原串变为目标元素删除最后一个元素,再插入最后一个元素。
  • 替换:dp[i][j]=dp[i-1][j-1]+rc。由原串除最后一个元素变为目标串除最后一个元素,再替换最后一个元素(特别地,若两串最后一个元素相等,则省去rc)

对空间进行压缩,空间复杂度压缩至O(min(M,N))

使用一维数组,由于dp[i][j]既依赖于dp[i][j-1]又依赖于dp[i-1][j-1],所以要有一个临时变量pre存dp[i-1][j-1].

注意

  • 注意对第一行和第一列的初始化。
  • 注意dp[i][j]表示的是不含strOri[i]和strTatget[j]元素的串,因为初始化第0行第0列的时候都表示的是空串。因此,开数组时也要多开一个位置。
  • 为了空间复杂度是两串长度的较短值:设置长串和短串,保证列数为短串的长度+1,如果目标串比原串长,则在换一下ic和+dc操作的代价值。

代码1(空间复杂度O(N)版,其中N为目标串长度)

import java.util.Scanner;
public class Main {
public static void main(String args[]) {
Scanner in=new Scanner(System.in);
while(in.hasNext()) {
String strOri=in.next();
String strTarget=in.next();
int ic=in.nextInt();
int dc=in.nextInt();
int rc=in.nextInt();
int editDis=getMinEditDis(strOri,strTarget,ic,dc,rc);
System.out.println(editDis);
}
} public static int getMinEditDis(String strOri,String strTarget,int ic,int dc,int rc){
int rows=strOri.length()+1;
int cols=strTarget.length()+1;
int dp[]=new int[cols]; for(int j=0;j<cols;++j) {
dp[j]=ic*j;//初始化第一行
} for(int i=1;i<rows;++i) {
int pre=dp[0];//pre记录dp[i-1][j-1]
dp[0]=dc*i;//初始化第一列
for(int j=1;j<cols;++j) {
int ansTemp;
if(strOri.charAt(i-1)==strTarget.charAt(j-1)) {//
ansTemp=pre;
}
else {
ansTemp=pre+rc;
}
int temp=dp[j];
dp[j]=getMin(ansTemp,dp[j]+dc,dp[j-1]+ic);//状态转移
pre=temp;//更新pre
}
}
return dp[cols-1];
} public static int getMin(int... vals) {
int min=Integer.MAX_VALUE;
for(int val:vals) {
if(val<min) {
min=val;
}
}
return min;
}
}

代码2(空间复杂度O(min(M,N))版)

import java.util.Scanner;

public class Main {
public static void main(String args[]) {
Scanner in=new Scanner(System.in);
while(in.hasNext()) {
String strOri=in.next();
String strTarget=in.next();
int ic=in.nextInt();
int dc=in.nextInt();
int rc=in.nextInt();
int editDis=getMinEditDis(strOri,strTarget,ic,dc,rc);
System.out.println(editDis);
}
} public static int getMinEditDis(String strOri,String strTarget,int ic,int dc,int rc){
String longs=strOri.length()>=strTarget.length()?strOri:strTarget;
String shorts=strOri.length()<strTarget.length()?strOri:strTarget;
int dp[]=new int[shorts.length()+1]; if(strOri.length()<strOri.length()) {
int temp=ic;
ic=dc;
dc=temp;
} for(int j=0;j<shorts.length()+1;++j) {
dp[j]=ic*j;//初始化第一行
} for(int i=1;i<longs.length()+1;++i) {
int pre=dp[0];//pre记录dp[i-1][j-1]
dp[0]=dc*i;//初始化第一列
for(int j=1;j<shorts.length()+1;++j) {
int ansTemp;
if(longs.charAt(i-1)==shorts.charAt(j-1)) {//
ansTemp=pre;
}
else {
ansTemp=pre+rc;
}
int temp=dp[j];
dp[j]=getMin(ansTemp,dp[j]+dc,dp[j-1]+ic);//状态转移
pre=temp;//更新pre
}
}
return dp[shorts.length()];
} public static int getMin(int... vals) {
int min=Integer.MAX_VALUE;
for(int val:vals) {
if(val<min) {
min=val;
}
}
return min;
}
}

[程序员代码面试指南]递归和动态规划-最小编辑代价(DP)的更多相关文章

  1. [程序员代码面试指南]递归和动态规划-数字字符串转换为字母组合的种数(DP)

    题意 给一个字符串,只由数字组成,若是'1'-'26',则认为可以转换为'a'-'z'对应的字母,问有多少种转换方法. 题解 状态转移很好想,注意dp多开一位,dp[0]为dp[2]的计算做准备.dp ...

  2. [程序员代码面试指南]递归和动态规划-排成一条线的纸牌博弈问题(DP)

    题目 给定一个整型数组arr,代表数值不同的纸牌排成一条线.玩家A和玩家B依次拿走每张纸牌,规定玩家A先拿,玩家B后拿,但是每个玩家每次只能拿走最左或最右的纸牌,玩家A和玩家B都绝顶聪明.请返回最后获 ...

  3. [程序员代码面试指南]递归和动态规划-最长公共子串问题(DP,LCST)

    问题描述 如题. 例:输入两个字符串 str1="1AB234",str2="1234EF" ,应输出最长公共子串"234". 解题思路 状 ...

  4. [程序员代码面试指南]递归和动态规划-换钱的方法数(DP,完全背包)

    题目描述 给定arr,arr中所有的值都为正数且不重复.每个值代表一种面值的货币,每种面值的货币可以使用任意张,再给定一个整数aim,求组成aim的方法数. 解题思路 完全背包 和"求换钱的 ...

  5. [程序员代码面试指南]递归和动态规划-换钱的最少货币数(DP,完全背包)

    题目描述 给定arr,arr中所有的值都为正数且不重复.每个值代表一种面值的货币,每种面值的货币可以使用任意张,再给定一个整数aim,求组成aim的最少货币数. 解题思路 dp[i][j]表示只用第0 ...

  6. [程序员代码面试指南]递归和动态规划-机器人达到指定位置方法数(一维DP待做)(DP)

    题目描述 一行N个位置1到N,机器人初始位置M,机器人可以往左/右走(只能在位置范围内),规定机器人必须走K步,最终到位置P.输入这四个参数,输出机器人可以走的方法数. 解题思路 DP 方法一:时间复 ...

  7. 程序员代码面试指南 IT名企算法与数据结构题目最优解

    原文链接 这是一本程序员面试宝典!书中对IT名企代码面试各类题目的最优解进行了总结,并提供了相关代码实现.针对当前程序员面试缺乏权威题目汇总这一痛点,本书选取将近200道真实出现过的经典代码面试题,帮 ...

  8. 程序员代码面试指南:IT名企算法与数据结构题目最优解

      第1章栈和队列 1设计一个有getMin功能的栈(士★☆☆☆) 1由两个栈组成的队列(尉★★☆☆) 5如何仅用递归函数和栈操作逆序一个栈(尉★★☆☆) 8猫狗队列(士★☆☆☆)10用一个栈实现另一 ...

  9. [程序员代码面试指南]二叉树问题-找到二叉树中的最大搜索二叉树(树形dp)

    题意 给定一颗二叉树的头节点,已知所有节点的值都不一样,找到含有节点最多的搜索二叉子树,并返回这个树的头节点. 题解 在后序遍历过程中实现. 求解步骤按树形dp中所列步骤.可能性三种:左子树最大.右子 ...

随机推荐

  1. 如何将返回的JSon字符串用MAP格式读取

    语法是这样: ObjectMapper mapper = new ObjectMapper(); Map resultMap=null; resultMap = mapper.readValue(in ...

  2. python:**kwargs

    **kwargs接收键值对参数,即字典, dict的pop()函数内需传2个参数,第一个参数为dict内的key, 如果有该key>第二个参数为None,最后的结果就是该key对应的value. ...

  3. Datawhale学数据分析第一章

    需要用到的基础知识pandas基础知识参考1,2章https://github.com/datawhalechina/joyful-pandas 1.导入数据tsv 制表符作为分隔符的字段符csv 逗 ...

  4. 你想了解的JDK 10版本更新都在这里

    「MoreThanJava」 宣扬的是 「学习,不止 CODE」,本系列 Java 基础教程是自己在结合各方面的知识之后,对 Java 基础的一个总回顾,旨在 「帮助新朋友快速高质量的学习」. 当然 ...

  5. nova 通过 python curl 创建虚拟机---keystone v2

    #! /bin/python #coding=utf- import urllib2 import json import requests # token post_url = 'http://12 ...

  6. 团队作业4:第一篇Scrum冲刺博客(歪瑞古德小队)

    目录 一.Alpha阶段任务认领 二.明日任务安排 三.项目预期任务量 四.敏捷开发前的感想 五.团队期望 Author:歪瑞古德小队 Project:海岛漂流 集合贴:团队作业4:项目冲刺集合贴(歪 ...

  7. packmol建模流程-计算

    一.建模流程(modelling procedure): 1.美国数据库下载amc.cif文件:http://rruff.geo.arizona.edu/AMS/amcsd.php 2.导入vesta ...

  8. 牛客网数据库SQL实战解析(41-50题)

    牛客网SQL刷题地址: https://www.nowcoder.com/ta/sql?page=0 牛客网数据库SQL实战解析(01-10题): https://blog.csdn.net/u010 ...

  9. 精讲响应式WebClient第6篇-请求失败自动重试机制,强烈建议你看一看

    本文是精讲响应式WebClient第6篇,前篇的blog访问地址如下: 精讲响应式webclient第1篇-响应式非阻塞IO与基础用法 精讲响应式WebClient第2篇-GET请求阻塞与非阻塞调用方 ...

  10. windows环境安装vue-cli及webpack并创建vueJs项目

    1. 安装node.js 2. 如果安装的是旧版本的 npm,可以通过 npm 命令来进行版本升级,命令如下: npm install npm -g npm网站服务器位于国外,所以经常下载缓慢或出现异 ...