力扣爆刷第132天之动态规划五连刷(子序列问题)

2024-05-05 07:20

本文主要是介绍力扣爆刷第132天之动态规划五连刷(子序列问题),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

力扣爆刷第132天之动态规划五连刷(子序列问题)

文章目录

      • 力扣爆刷第132天之动态规划五连刷(子序列问题)
      • 总结:
      • 一、1035. 不相交的线
      • 二、53. 最大子数组和
      • 三、392. 判断子序列
      • 四、115. 不同的子序列
      • 五、583. 两个字符串的删除操作

总结:

本节的题目均是上一节四种类型的变体。

一、1035. 不相交的线

题目链接:https://leetcode.cn/problems/uncrossed-lines/description/
思路:本题是子序列问题的变体,求不相交的线的最大个数,其实就是求最长重复子序列,求子序列,不等问题也需要处理。
定义dp[i][j]表示以nums1[i]和nums2[j]为结尾的区间中最长重复子序列的长度。
那么根据定义:
nums1[i] = nums2[j],即可得 dp[i][j] = dp[i-1][j-1]+1。
nums1[i] != nums2[j],即可得 dp[i][j] = max(dp[i][j-1], dp[i-1][j])。

class Solution {public int maxUncrossedLines(int[] nums1, int[] nums2) {int[][] dp = new int[nums1.length+1][nums2.length+1];for(int i = 0; i < nums1.length; i++) {for(int j = 0; j < nums2.length; j++) {if(nums1[i] == nums2[j]) {dp[i+1][j+1] = dp[i][j] + 1;}else{dp[i+1][j+1] = Math.max(dp[i+1][j], dp[i][j+1]);}}}return dp[nums1.length][nums2.length];}
}

二、53. 最大子数组和

题目链接:https://leetcode.cn/problems/maximum-subarray/description/
思路:求最大子数组的和,其实就是最长连续子数组的和,是这个类型的变体,定义dp[i]表示以nums[i]为结尾的子序列和的最大值,如果前面子序列的和加上当前元素,结果还比当前元素小,就没必要加了。
另外的结题思路就是贪心,贪心的思路就是一直计算累加和,并且记录最大值,只要累加和小于0了,就从新计算累加和。

class Solution {public int maxSubArray(int[] nums) {int[] dp = new int[nums.length];dp[0] = nums[0];int max = nums[0];for(int i = 1; i < nums.length; i++) {if(dp[i-1] + nums[i] > nums[i]) {dp[i] = dp[i-1] + nums[i];}else{dp[i] = nums[i];}max = Math.max(max, dp[i]);}return max;}
}

三、392. 判断子序列

题目链接:https://leetcode.cn/problems/is-subsequence/description/
思路:求一个字符串是否是另一个字符串的子序列,其实求的就是最长相等子序列。本题可以用贪心,也可以用动规。
贪心的话就是,遍历长字符串,只要短字符串中有想等的,短字符串就往前走一步。
动规的话,定义dp[i][j]表示以下标i为结尾的字符串s,是否是以下标j为结尾的字符串t的字符子串。所以元素相等时依赖于上一个位置,元素不等时,s不可后退,t可后退。

class Solution {public boolean isSubsequence(String s, String t) {int k = 0, j = 0;for(int i = 0; i < t.length() && j < s.length(); i++) {if(s.charAt(j) == t.charAt(i)) {k++;j++;}}return k == s.length();}
}

动态规划

class Solution {public boolean isSubsequence(String s, String t) {if(s.length() > t.length()) return false;boolean[][] dp = new boolean[s.length()+1][t.length()+1];Arrays.fill(dp[0], true);for(int i = 0; i < s.length(); i++) {for(int j = 0; j < t.length(); j++) {if(s.charAt(i) == t.charAt(j)) {dp[i+1][j+1] = dp[i][j];}else{dp[i+1][j+1] = dp[i+1][j];}}}return dp[s.length()][t.length()];}
}

四、115. 不同的子序列

题目链接:https://leetcode.cn/problems/distinct-subsequences/description/
思路:本题和上一题类似,上一题是s不能后退,t可以,因为求的是完整的s。本题是t不能后退,s可以,因为求的是完整的t。
求不同子序列,定义dp[i][j]表示,以下标i为结尾的字符串包含dp[i][j]个以j为结尾的t。
当s[i] = t[j]时,求组合数,dp[i][j] = dp[i-1][j-1] + dp[i-1][j]。 如abb 和 ab。i= 2和j=1时可以体会一下。
当s[i]!=t[j]时,求组合数,dp[i][j] = dp[i-1][j]。

class Solution {public int numDistinct(String s, String t) {int[][] dp = new int[s.length()+1][t.length()+1];for(int i = 0; i < dp.length; i++) {dp[i][0] = 1;}for(int i = 0; i < s.length(); i++) {for(int j = 0; j < t.length(); j++) {if(s.charAt(i) == t.charAt(j)) {dp[i+1][j+1] = dp[i][j] + dp[i][j+1];}else{dp[i+1][j+1] = dp[i][j+1];}}}return dp[s.length()][t.length()];}
}

五、583. 两个字符串的删除操作

题目链接:https://leetcode.cn/problems/delete-operation-for-two-strings/description/
思路:本题是求最少删除多少次,可以让两个字符串相等。定义也是一样的,定义dp[i][j]表示以i为为结尾的字符串word1和以j为结尾的字符串word2,需要dp[i][j]此删除操作才相等。
那么当word1[i] == word2[j]时,就无需删除,依赖于上一个位置要删除的个数,dp[i][j] = dp[i-1][j-1]。
当word1[i] != word2[j] 时,就需要考虑删除,可以考虑是删掉Word1一个还是删掉word2一个,反正就是最少个数。dp[i][j] = min(dp[i-1][j]), dp[i][j-1] + 1;

class Solution {public int minDistance(String word1, String word2) {int[][] dp = new int[word1.length()+1][word2.length()+1];for(int i = 0; i < dp.length; i++) {dp[i][0] = i;}for(int i = 0; i < dp[0].length; i++) {dp[0][i] = i;}for(int i = 0; i < word1.length(); i++) {for(int j = 0; j < word2.length(); j++) {if(word1.charAt(i) == word2.charAt(j)) {dp[i+1][j+1] = dp[i][j];}else{dp[i+1][j+1] = Math.min(dp[i+1][j], dp[i][j+1]) + 1;}}}return dp[word1.length()][word2.length()];}
}

这篇关于力扣爆刷第132天之动态规划五连刷(子序列问题)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/961113

相关文章

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

numpy求解线性代数相关问题

《numpy求解线性代数相关问题》本文主要介绍了numpy求解线性代数相关问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 在numpy中有numpy.array类型和numpy.mat类型,前者是数组类型,后者是矩阵类型。数组

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

vue解决子组件样式覆盖问题scoped deep

《vue解决子组件样式覆盖问题scopeddeep》文章主要介绍了在Vue项目中处理全局样式和局部样式的方法,包括使用scoped属性和深度选择器(/deep/)来覆盖子组件的样式,作者建议所有组件... 目录前言scoped分析deep分析使用总结所有组件必须加scoped父组件覆盖子组件使用deep前言

VUE动态绑定class类的三种常用方式及适用场景详解

《VUE动态绑定class类的三种常用方式及适用场景详解》文章介绍了在实际开发中动态绑定class的三种常见情况及其解决方案,包括根据不同的返回值渲染不同的class样式、给模块添加基础样式以及根据设... 目录前言1.动态选择class样式(对象添加:情景一)2.动态添加一个class样式(字符串添加:情

解决Cron定时任务中Pytest脚本无法发送邮件的问题

《解决Cron定时任务中Pytest脚本无法发送邮件的问题》文章探讨解决在Cron定时任务中运行Pytest脚本时邮件发送失败的问题,先优化环境变量,再检查Pytest邮件配置,接着配置文件确保SMT... 目录引言1. 环境变量优化:确保Cron任务可以正确执行解决方案:1.1. 创建一个脚本1.2. 修

Python 标准库time时间的访问和转换问题小结

《Python标准库time时间的访问和转换问题小结》time模块为Python提供了处理时间和日期的多种功能,适用于多种与时间相关的场景,包括获取当前时间、格式化时间、暂停程序执行、计算程序运行时... 目录模块介绍使用场景主要类主要函数 - time()- sleep()- localtime()- g