【LeetCode】动态规划—712. 两个字符串的最小ASCII删除和(附完整Python/C++代码)
动态规划—712. 两个字符串的最小ASCII删除和
- 前言
- 题目描述
- 基本思路
- 1. 问题定义
- 2. 理解问题和递推关系
- 3. 解决方法
- 3.1 动态规划方法
- 3.2 空间优化的动态规划
- 4. 进一步优化
- 5. 小总结
- 代码实现
- Python
- Python3代码实现
- Python 代码解释
- C++
- C++代码实现
- C++ 代码解释
- 总结:
前言
在字符串处理的过程中,如何有效地将两个字符串转换为相同的形式是一个重要的问题。最小 ASCII 删除和问题提供了一种评估字符串相似性的有效方法,通过计算所需删除字符的 ASCII 值和,为我们提供了清晰的转换成本。本文将探讨这一问题的基本思路,并给出动态规划的实现方法,最后展示 Python 和 C++ 的具体代码。
题目描述

基本思路
1. 问题定义
最小 ASCII 删除和问题要求我们找出将两个字符串 s 1 s 1 s1 和 s 2 s 2 s2 转换为相同字符串所需删除的字符的最小 ASCII 值之和。换句话说,计算出为了使两个字符串相同,所需删除的字符的 ASCII 值的总和。
2. 理解问题和递推关系
- 对于两个字符串 s 1 s 1 s1 和 s 2 s 2 s2 ,我们可以定义
dp[i][j]为将 s 1 s 1 s1 的前 i i i 个字符和 s 2 s 2 s2 的前 j个字符变为相同的最小 ASCII 删除和。 - 递推关系如下:
- 如果 s 1 [ i − 1 ] = = s 2 [ j − 1 ] s 1[i-1]==s 2[j-1] s1[i−1]==s2[j−1] ,那么不需要删除任何字符, d p [ i ] [ j ] = d p [ i − 1 ] [ j − 1 ] d p[i][j]=d p[i-1][j-1] dp[i][j]=dp[i−1][j−1] 。
- 如果 s 1 [ i − 1 ] ! = s 2 [ j − 1 ] s 1[i-1]!=s 2[j-1] s1[i−1]!=s2[j−1], 则有三种情况:
- 删除 s 1 [ i − 1 ] s1[i-1] s1[i−1],代价为 ord ( s 1 [ i − 1 ] ) + d p [ i − 1 ] [ j ] \operatorname{ord}(s 1[i-1])+d p[i-1][j] ord(s1[i−1])+dp[i−1][j] 。
- 删除 s 2 [ j − 1 ] s 2[j-1] s2[j−1] ,代价为 ord ( s 2 [ j − 1 ] ) + d p [ i ] [ j − 1 ] \operatorname{ord}(s 2[j-1])+d p[i][j-1] ord(s2[j−1])+dp[i][j−1] 。
- 同时删除 s 1 [ i − 1 ] s 1[i-1] s1[i−1] 和 s 2 [ j − 1 ] s 2[j-1] s2[j−1] ,代价为 ord ( s 1 [ i − 1 ] ) + ord ( s 2 [ j − 1 ] ) + dp [ i − 1 ] [ j − \operatorname{ord}(s 1[i-1])+\operatorname{ord}(s 2[j-1])+\operatorname{dp}[i-1][j- ord(s1[i−1])+ord(s2[j−1])+dp[i−1][j− 1].
- 因此,综合以上情况:
d p [ i ] [ j ] = min ( d p [ i − 1 ] [ j ] + ord ( s 1 [ i − 1 ] ) , d p [ i ] [ j − 1 ] + ord ( s 2 [ j − 1 ] ) , d p [ i − 1 ] [ j − 1 ] + ord ( s 1 [ i − 1 ] ) + ord ( s 2 [ j − 1 ] ) ) d p[i][j]=\min (d p[i-1][j]+\operatorname{ord}(s 1[i-1]), d p[i][j-1]+\operatorname{ord}(s 2[j-1]), d p[i-1][j-1]+\operatorname{ord}(s 1[i-1])+\operatorname{ord}(s 2[j-1])) dp[i][j]=min(dp[i−1][j]+ord(s1[i−1]),dp[i][j−1]+ord(s2[j−1]),dp[i−1][j−1]+ord(s1[i−1])+ord(s2[j−1]))
3. 解决方法
3.1 动态规划方法
- 创建一个二维数组 d p d p dp ,大小为 ( m + 1 ) × ( n + 1 ) (m+1) \times(n+1) (m+1)×(n+1) ,其中 m m m 和 n n n 分别是 s 1 s 1 s1 和 s 2 s 2 s2 的长度。
- 初始化边界条件:
- d p [ i ] [ 0 ] = ∑ k = 0 i − 1 ord ( s 1 [ k ] ) dp[i][0]=\sum_{k=0}^{i-1} \text{ord}(s1[k]) dp[i][0]=∑k=0i−1ord(s1[k]),表示将 s 1 s 1 s1 的前 i i i 个字符转换为空字符串所需删除的 ASCII 值之和。
- d p [ 0 ] [ j ] = ∑ k = 0 j − 1 ord ( s 2 [ k ] ) dp[0][j]=\sum_{k=0}^{j-1} \text{ord}(s2[k]) dp[0][j]=∑k=0j−1ord(s2[k]),表示将 s 2 s 2 s2 的前 j j j 个字符转换为空字符串所需删除的 ASCII 值之和。
- 使用双重石环填充 dp 数组,依赖于前面的状态。
- 最终结果为 d p [ m ] [ n ] \mathrm{dp}[\mathrm{m}][\mathrm{n}] dp[m][n] 。
3.2 空间优化的动态规划
- 可以使用一维数组来优化空间复杂度,减少内存占用。
4. 进一步优化
通过空间优化,降低内存占用的同时保持时间复杂度为 O ( m ∗ n ) O(m * n) O(m∗n),适合中等规模的字符串比较。
5. 小总结
- 最小 ASCII 删除和问题通过动态规划有效地解决了两个字符串之间的转换成本。
- 该问题的解法展示了如何设计状态转移方程,并且可以通过空间优化提高性能。
- 理解该问题不仅有助于掌握动态规划的应用,还为处理相似问题提供了思路。
以上就是两个字符串的最小ASCII删除和问题的基本思路。
代码实现
Python
Python3代码实现
class Solution:def minimumDeleteSum(self, s1: str, s2: str) -> int:m, n = len(s1), len(s2)# 创建dp数组dp = [[0] * (n + 1) for _ in range(m + 1)]# 初始化边界条件for i in range(1, m + 1):dp[i][0] = dp[i - 1][0] + ord(s1[i - 1]) # 删除s1的字符for j in range(1, n + 1):dp[0][j] = dp[0][j - 1] + ord(s2[j - 1]) # 删除s2的字符# 填充dp数组for i in range(1, m + 1):for j in range(1, n + 1):if s1[i - 1] == s2[j - 1]:dp[i][j] = dp[i - 1][j - 1] # 字符相同else:dp[i][j] = min(dp[i - 1][j] + ord(s1[i - 1]), # 删除s1的字符dp[i][j - 1] + ord(s2[j - 1]), # 删除s2的字符dp[i - 1][j - 1] + ord(s1[i - 1]) + ord(s2[j - 1])) # 同时删除# 返回最小ASCII删除和return dp[m][n]
Python 代码解释
- 初始化:创建
dp数组并设置边界条件,分别表示将s1和s2转换为空字符串的操作。 - 填充 dp 数组:使用双重循环计算每个子问题的最小 ASCII 删除和,依赖于之前的结果。
- 返回结果:最终返回
dp[m][n],即将s1转换为s2所需的最小 ASCII 删除和。
C++
C++代码实现
class Solution {
public:int minimumDeleteSum(string s1, string s2) {int m = s1.size(), n = s2.size();// 创建dp数组vector<vector<int>> dp(m + 1, vector<int>(n + 1, 0));// 初始化边界条件for (int i = 1; i <= m; i++) {dp[i][0] = dp[i - 1][0] + s1[i - 1]; // 删除s1的字符}for (int j = 1; j <= n; j++) {dp[0][j] = dp[0][j - 1] + s2[j - 1]; // 删除s2的字符}// 填充dp数组for (int i = 1; i <= m; i++) {for (int j = 1; j <= n; j++) {if (s1[i - 1] == s2[j - 1]) {dp[i][j] = dp[i - 1][j - 1]; // 字符相同} else {dp[i][j] = min({dp[i - 1][j] + s1[i - 1], // 删除s1的字符dp[i][j - 1] + s2[j - 1], // 删除s2的字符dp[i - 1][j - 1] + s1[i - 1] + s2[j - 1]}); // 同时删除}}}// 返回最小ASCII删除和return dp[m][n];}
};
C++ 代码解释
- 初始化:创建
dp数组并设置边界条件,分别表示将s1和s2转换为空字符串的操作。 - 动态规划填充:使用双重循环遍历每个可能的子问题,依据字符是否相同来更新 dp 数组。
- 返回结果:返回
dp[m][n],即将s1转换为s2所需的最小 ASCII 删除和。
总结:
- 最小 ASCII 删除和问题通过动态规划有效地解决了字符串之间的转换成本,具有广泛的实际应用。
- 理解并掌握该问题的解法,不仅对学习动态规划有帮助,还为处理其他类似问题提供了思路。
相关文章:
【LeetCode】动态规划—712. 两个字符串的最小ASCII删除和(附完整Python/C++代码)
动态规划—712. 两个字符串的最小ASCII删除和 前言题目描述基本思路1. 问题定义2. 理解问题和递推关系3. 解决方法3.1 动态规划方法3.2 空间优化的动态规划 4. 进一步优化5. 小总结 代码实现PythonPython3代码实现Python 代码解释 CC代码实现C 代码解释 总结: 前言 在字符串处…...
wordpress Contact Form 7插件提交留言时发生错误可能的原因
WordPress Contact Form 7 插件提交留言时发生错误可能有以下几种原因,并提供相应的解决方案: 1. 表单字段验证失败 原因: 用户输入的数据未通过表单字段的验证规则。 解决方案: – 检查表单字段的验证规则是否设置正确。 –…...
uibot发送邮件:自动化邮件发送教程详解!
uibot发送邮件的操作指南?uibot发送邮件的两种方式? 在现代办公环境中,自动化流程的引入极大地提高了工作效率。uibot发送邮件功能成为了许多企业和个人实现邮件自动化发送的首选工具。AokSend将详细介绍如何使用uibot发送邮件。 uibot发送…...
【PostgreSQL】PG数据库表“膨胀”粗浅学习
文章目录 1 为什么需要关注表膨胀?2 如何确定是否发生了表膨胀?2.1 通过查询表的死亡元组占比情况来判断膨胀率2.1.1 指定数据库和表名2.1.2 查询数据库里面所有表的膨胀情况 3 膨胀的原理3.1 什么是膨胀?膨胀率?3.2 哪些数据库元…...
力扣(leetcode)每日一题 871 最低加油次数 | 贪心
871. 最低加油次数 题干 汽车从起点出发驶向目的地,该目的地位于出发位置东面 target 英里处。 沿途有加油站,用数组 stations 表示。其中 stations[i] [positioni, fueli] 表示第 i 个加油站位于出发位置东面 positioni 英里处,并且有 f…...
ppt压缩文件怎么压缩?压缩PPT文件的多种压缩方法
ppt压缩文件怎么压缩?当文件体积过大时,分享和传输就会变得困难。许多电子邮件服务对附件的大小有限制,而在网络环境不佳时,上传和下载大文件可能耗时较长。此外,在不同设备上播放时,较大的PPT文件还可能导…...
2024.10月11日--- SpringMVC拦截器
拦截器 1 回顾过滤器: Servlet规范中的三大接口:Servlet接口,Filter接口、Listener接口。 过滤器接口,是Servlet2.3版本以来,定义的一种小型的,可插拔的Web组件,可以用来拦截和处理Servlet容…...
uniapp 锁屏显示插件 Ba-LockShow(可让vue直接具备锁屏显示能力)
简介 Ba-LockShow 是一款可以直接使uniapp的vue界面在锁屏页展示的插件。 支持使vue直接具备锁屏显示能力支持设置锁屏显示和不显示支持唤醒屏幕 截图展示(仅参考) 支持定制、本地包、源码等,有建议和需要,请点击文章结尾“Unia…...
CSS计数器
CSS 中的计数器类似于变量,可以实现简单的计数功能,并将结果显示在页面上,在早期的网站上应用比较广泛。要实现计数器需要用到以下几个属性: counter-reset:创建或者重置计数器;counter-increment…...
嵌入式Linux:信号集
目录 1、信号集初始化 2、向信号集中添加或删除信号 3、测试信号是否在信号集中 在 Linux 系统中,处理多个信号时常用到一种数据结构:信号集(sigset_t)。信号集允许我们将多个信号组织在一起,以便在系统调用中传递和…...
Linux 外设驱动 应用 1 IO口输出
从这里开始外设驱动介绍,这里使用的IMX8的芯片作为驱动介绍 开发流程: 修改设备树,配置 GPIO1_IO07 为 GPIO 输出。使用 sysfs 接口或编写驱动程序控制 GPIO 引脚。编译并测试。 这里假设设备树,已经配置好了。不在论述这个问题…...
基于SpringBoot+Vue+MySQL的留守儿童爱心网站
系统展示 用户前台界面 管理员后台界面 系统背景 随着现代社会的发展,留守儿童问题日益受到关注。传统的纸质管理方式已经无法满足现代人们对留守儿童爱心信息的需求。为了提高留守儿童爱心信息的管理效率,增加用户信息的安全性,并方便及时反…...
调用第三方接口
目录 一、分析给出的接口文档 二、请求体格式之间的区别 三、示例代码 一、分析给出的接口文档 一般的接口文档包括以下几大部分: 1、请求URL:http://{ip}:{port}/api/ec/dev/message/sendCustomMessageSingle 2、请求方式:POST、GET等 3、…...
JAVA 多线程入门例子:CountDownLatch
首先确定线程数量。如果数据集合的大小小于50,就只使用一个线程;否则使用5个线程。计算每个线程平均处理的数据数量sizePerThread以及余数remainder。在划分数据子集合时,对于每个线程的处理范围进行计算。如果有余数,就将余数依次…...
k8s jenkins 动态创建slave
k8s jenkins 动态创建slave 简述使用jenkins动态slave的优势:配置jenkins动态slave配置 Pod Template配置容器模板挂载卷 测试 简述 持续构建与发布是我们日常工作中必不可少的一个步骤,目前大多公司都采用 Jenkins 集群来搭建符合需求的 CI/CD 流程&am…...
MVS海康工业相机达不到标称最大帧率
文章目录 一、相机参数设置1、取消相机帧率限制2、修改相机图像格式3、调整相机曝光时间4、检查相机数据包大小(网口相机特有参数)5、 恢复相机默认参数6、 相机 ADC 输出位深调整 二、系统环境设置1、 网口相机设置2、 USB 相机设置 一、相机参数设置 …...
数据结构:用双栈实现一个队列
要用两个栈实现一个队列,可以利用“栈”的后进先出 (LIFO) 特性来模拟“队列”的先进先出 (FIFO) 操作。具体做法是使用两个栈:一个作为入栈栈,另一个作为出栈栈。 算法步骤 入队操作(enqueue): 将元素压…...
QScroller Class
Header:#include < QScroller > qmake:QT += widgets Since:Qt 5.0 Inherits:QObject This class was introduced in Qt 5.0. Public Types enum Input {InputPress, InputMove, InputRelease } enum ScrollerGestureType {TouchGesture, LeftMouseButtonGesture,…...
React高阶组件详解
React高阶组件(HOC)详解 定义 React高阶组件(HOC)是一个函数,该函数接受一个组件作为参数并返回一个新的组件。高阶组件本身不是一个组件,而是一个函数,它利用React的组合特性,对传入…...
TextView把其它控件挤出屏幕的处理办法
1.如果TextView后面的控件是紧挨着TextView的,可以给TextView添加maxWidth限制其最大长度 上有问题的布局代码 <?xml version"1.0" encoding"utf-8"?> <layout xmlns:android"http://schemas.android.com/apk/res/android&qu…...
手游刚开服就被攻击怎么办?如何防御DDoS?
开服初期是手游最脆弱的阶段,极易成为DDoS攻击的目标。一旦遭遇攻击,可能导致服务器瘫痪、玩家流失,甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案,帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...
设计模式和设计原则回顾
设计模式和设计原则回顾 23种设计模式是设计原则的完美体现,设计原则设计原则是设计模式的理论基石, 设计模式 在经典的设计模式分类中(如《设计模式:可复用面向对象软件的基础》一书中),总共有23种设计模式,分为三大类: 一、创建型模式(5种) 1. 单例模式(Sing…...
C++初阶-list的底层
目录 1.std::list实现的所有代码 2.list的简单介绍 2.1实现list的类 2.2_list_iterator的实现 2.2.1_list_iterator实现的原因和好处 2.2.2_list_iterator实现 2.3_list_node的实现 2.3.1. 避免递归的模板依赖 2.3.2. 内存布局一致性 2.3.3. 类型安全的替代方案 2.3.…...
从WWDC看苹果产品发展的规律
WWDC 是苹果公司一年一度面向全球开发者的盛会,其主题演讲展现了苹果在产品设计、技术路线、用户体验和生态系统构建上的核心理念与演进脉络。我们借助 ChatGPT Deep Research 工具,对过去十年 WWDC 主题演讲内容进行了系统化分析,形成了这份…...
云启出海,智联未来|阿里云网络「企业出海」系列客户沙龙上海站圆满落地
借阿里云中企出海大会的东风,以**「云启出海,智联未来|打造安全可靠的出海云网络引擎」为主题的阿里云企业出海客户沙龙云网络&安全专场于5.28日下午在上海顺利举办,现场吸引了来自携程、小红书、米哈游、哔哩哔哩、波克城市、…...
UE5 学习系列(三)创建和移动物体
这篇博客是该系列的第三篇,是在之前两篇博客的基础上展开,主要介绍如何在操作界面中创建和拖动物体,这篇博客跟随的视频链接如下: B 站视频:s03-创建和移动物体 如果你不打算开之前的博客并且对UE5 比较熟的话按照以…...
基于数字孪生的水厂可视化平台建设:架构与实践
分享大纲: 1、数字孪生水厂可视化平台建设背景 2、数字孪生水厂可视化平台建设架构 3、数字孪生水厂可视化平台建设成效 近几年,数字孪生水厂的建设开展的如火如荼。作为提升水厂管理效率、优化资源的调度手段,基于数字孪生的水厂可视化平台的…...
【服务器压力测试】本地PC电脑作为服务器运行时出现卡顿和资源紧张(Windows/Linux)
要让本地PC电脑作为服务器运行时出现卡顿和资源紧张的情况,可以通过以下几种方式模拟或触发: 1. 增加CPU负载 运行大量计算密集型任务,例如: 使用多线程循环执行复杂计算(如数学运算、加密解密等)。运行图…...
Spring AI与Spring Modulith核心技术解析
Spring AI核心架构解析 Spring AI(https://spring.io/projects/spring-ai)作为Spring生态中的AI集成框架,其核心设计理念是通过模块化架构降低AI应用的开发复杂度。与Python生态中的LangChain/LlamaIndex等工具类似,但特别为多语…...
.Net Framework 4/C# 关键字(非常用,持续更新...)
一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...
