LeetCode算法题解(动态规划)|LeetCode1143. 最长公共子序列、LeetCode1035. 不相交的线、LeetCode53. 最大子数组和

一、LeetCode1143. 最长公共子序列

题目链接:1143. 最长公共子序列
题目描述:

给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 ,返回 0 。

一个字符串的 子序列 是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何字符)后组成的新字符串。

  • 例如,"ace" 是 "abcde" 的子序列,但 "aec" 不是 "abcde" 的子序列。

两个字符串的 公共子序列 是这两个字符串所共同拥有的子序列。

示例 1:

输入:text1 = "abcde", text2 = "ace" 
输出:3  
解释:最长公共子序列是 "ace" ,它的长度为 3 。

示例 2:

输入:text1 = "abc", text2 = "abc"
输出:3
解释:最长公共子序列是 "abc" ,它的长度为 3 。

示例 3:

输入:text1 = "abc", text2 = "def"
输出:0
解释:两个字符串没有公共子序列,返回 0 。

提示:

  • 1 <= text1.length, text2.length <= 1000
  • text1 和 text2 仅由小写英文字符组成。
算法分析:
定义dp数组及下标含义:

dp[i][j]表示在区间[0,i]之间的text1字符串,与区间[0,j]的text2字符串最长的公共子序列长度。

递推公式:

如果字符text[i]与字符text2[j]相等,那么dp[i][j]可由dp[i-1][j-1]推出来,即[0,i]之间的text1字符串与[0,j]之间text2字符串的最长公共子序列长度等于,[0,i-1]之间的text1字符串与[0,j-1]之间的text2字符串的最长公共子序列长度加一。

如果不相等,那么dp[i][j]可由两个方向推导出来:

1、[0,i-1]之间的text1字符串与[0,j]之间的text2字符串的最长公共子序列长度。

2、[0,i]之间的text1字符串与[0,j-1]之间的text2字符串的最长公共子序列长度。

所以dp[i][j]=max(dp[i-1][j],dp[i][j-1]);

初始化:

因为dp[i][j]是由dp[i-1][j-1],dp[i-1][j],dp[i][j-1]三个方向推出来的,所以需要初始化最左边一列和最上边那一行。

        for(int i = 0; i < text2.length(); i++){//如果dp[0][i]被初始化成1,那么后面的都需初始化成1if(text1.charAt(0) == text2.charAt(i)){dp[0][i] = 1;}else if(i == 0){dp[0][i] = 0;}else{dp[0][i] = dp[0][i-1];}}for(int i = 0; i < text1.length(); i++){//如果dp[i][0]被初始化成1,那么后面的都需初始化成1if(text1.charAt(i) == text2.charAt(0)){dp[i][0] = 1;}else if(i == 0){dp[i][0] = 0;}else{dp[i][0] = dp[i-1][0];}}
遍历顺序:

先遍历text1再遍历text2或先遍历text2再遍历text1都可以,不过都需从下标1开始往后遍历。

打印dp数组进行验证。

代码如下:

class Solution {public int longestCommonSubsequence(String text1, String text2) {int[][] dp = new int[text1.length()][text2.length()];for(int i = 0; i < text2.length(); i++){if(text1.charAt(0) == text2.charAt(i)){dp[0][i] = 1;}else if(i == 0){dp[0][i] = 0;}else{dp[0][i] = dp[0][i-1];}}for(int i = 0; i < text1.length(); i++){if(text1.charAt(i) == text2.charAt(0)){dp[i][0] = 1;}else if(i == 0){dp[i][0] = 0;}else{dp[i][0] = dp[i-1][0];}}for(int i = 1; i < text1.length(); i++) {for(int j = 1; j < text2.length(); j++) {if(text1.charAt(i) == text2.charAt(j)){dp[i][j] = dp[i-1][j-1]+1;}else{dp[i][j] = Math.max(dp[i-1][j], dp[i][j-1]);}}}// for(int i = 0; i < text1.length(); i++) {//     for(int j = 0; j < text2.length(); j++) {//         System.out.print(dp[i][j]+" ");//     }//     System.out.println();// }return dp[text1.length()-1][text2.length()-1];}
}

二、LeetCode1035. 不相交的线

题目链接:1035. 不相交的线
题目描述:

在两条独立的水平线上按给定的顺序写下 nums1 和 nums2 中的整数。

现在,可以绘制一些连接两个数字 nums1[i] 和 nums2[j] 的直线,这些直线需要同时满足满足:

  •  nums1[i] == nums2[j]
  • 且绘制的直线不与任何其他连线(非水平线)相交。

请注意,连线即使在端点也不能相交:每个数字只能属于一条连线。

以这种方法绘制线条,并返回可以绘制的最大连线数。

示例 1:

输入:nums1 = [1,4,2], nums2 = [1,2,4]
输出:2
解释:可以画出两条不交叉的线,如上图所示。 
但无法画出第三条不相交的直线,因为从 nums1[1]=4 到 nums2[2]=4 的直线将与从 nums1[2]=2 到 nums2[1]=2 的直线相交。

示例 2:

输入:nums1 = [2,5,1,2,5], nums2 = [10,5,2,1,5,2]
输出:3

示例 3:

输入:nums1 = [1,3,7,1,7,5], nums2 = [1,9,2,5,1]
输出:2

提示:

  • 1 <= nums1.length, nums2.length <= 500
  • 1 <= nums1[i], nums2[j] <= 2000
算法分析:

这道题的思路其实跟上一道题是一模一样的,要求可以绘制的不相交的最大连线数,也就是求两个数组的最大公共子序列。

定义dp数组及下标含义:

dp[i][j]表示[0,i]区间的nums1子数组和[0,j]区间的nums2子数组的最长公共子序列。

递推公式:

如果nums1[i]与nums2[j]相等,那么dp[i][j]=dp[i-1][j-1]+1;

如果不相等那么dp[i][j]=max(dp[i-1][j],dp[i][j-1]);

初始化:

初始化第一列和第一行。

        for(int i = 0; i < nums1.length; i++) {if(nums1[i] == nums2[0]){dp[i][0] = 1;}else if(i == 0){dp[i][0] = 0;}else{dp[i][0] = dp[i-1][0];}}for(int i = 0; i < nums2.length; i++) {if(nums2[i] == nums1[0]){dp[0][i] = 1;}else if(i == 0){dp[0][i] = 0;}else{dp[0][i] = dp[0][i-1];}}
遍历顺序:

先遍历nums1在遍历nums2或先遍历nums2在遍历nums1都可以。

打印dp数组验证。

代码如下:

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

三、LeetCode53. 最大子数组和

题目链接:53. 最大子数组和
题目描述:

给你一个整数数组 nums ,请你找出一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

子数组 是数组中的一个连续部分。

示例 1:

输入:nums = [-2,1,-3,4,-1,2,1,-5,4]
输出:6
解释:连续子数组 [4,-1,2,1] 的和最大,为 6 。

示例 2:

输入:nums = [1]
输出:1

示例 3:

输入:nums = [5,4,-1,7,8]
输出:23

提示:

  • 1 <= nums.length <= 105
  • -104 <= nums[i] <= 104
算法分析:
定义dp数组及下标含义:

dp[i]表示以nums[i]结尾的子序列的最大和。

递推公式:

dp[i]可以由两个方向推出来,一个是dp[i-1]+nums[i],以nums[i-1]结束的子序列的最大和加上第i个元素;两一个是nums[i],重新开始一段子序列和。

初始化:

dp[0]=nums[0]。

遍历顺序:

从前往后遍历数组中的元素。

打印dp数组。

代码如下:

class Solution {public int maxSubArray(int[] nums) {int[] dp = new int[nums.length];dp[0] = nums[0];int result = dp[0];//记录最大的子数组和。for(int i = 1; i < nums.length; i++){dp[i] = Math.max(dp[i-1] + nums[i], nums[i]);if(dp[i] > result) result = dp[i];}return result;}
}

总结

前两道题很类似,只要会其中一道题,另外一道题也会迎刃而解。

第三题暴力算的话会超时,动规也不容易想出来。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/208447.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

GitHub----使用记录

一、上传文件到仓库 1、首先新建一个github仓库 然后先记住这一句指令 2、下载git工具 https://git-scm.com/downloads 下载工具安装不用运行 3、使用git工具上传文件并推送 找到你想上传的文件的位置&#xff0c;右击git Bush here git init &#xff1a;初始化这个仓…

canvas基础:绘制贝塞尔曲线

canvas实例应用100 专栏提供canvas的基础知识&#xff0c;高级动画&#xff0c;相关应用扩展等信息。 canvas作为html的一部分&#xff0c;是图像图标地图可视化的一个重要的基础&#xff0c;学好了canvas&#xff0c;在其他的一些应用上将会起到非常重要的帮助。 文章目录 bez…

Xshell连接VMware虚拟机中的CentOS

Xshell连接VMware虚拟机中的CentOShttps://www.cnblogs.com/niuben/p/13157291.html 步骤&#xff1a; 1. 检查Linux虚拟机的网络连接模式&#xff0c;确保它是NAT模式。&#xff08;由于只在本机进行连接&#xff0c;所以没有选择桥接模式。当然&#xff0c;桥接模式的配置会…

Python函数的高级用法

Python 的函数是“一等公民”&#xff0c;因此函数本身也是一个对象&#xff0c;函数既可用于赋值&#xff0c;也可用作其他函数的参数&#xff0c;还可作为其他函数的返回值。 使用函数变量 Python 的函数也是一种值&#xff1a;所有函数都是 function 对象&#xff0c;这意…

算法学习系列(三):汉诺塔

目录&#xff1a; 引言一、问题描述二、问题求解三、测试四、附录&#xff08;所有代码&#xff09; 引言 这个汉诺塔问题就是一个典型的递归问题&#xff0c;这篇博客也算是上一篇的一个扩展吧&#xff0c;都是递归问题&#xff0c;这个问题太大&#xff0c;而且牵扯到的问题…

第一类瑞利索末菲标量衍射模型的方孔衍射的空间像计算(附python计算代码)

记第一类瑞利索末菲标量衍射模型的方孔衍射的空间像计算(附python计算代码) RS type 1 衍射空间像计算傅里叶变换采样条件实际计算计算要求傅立叶变换法计算直接卷积方法计算代码傅立叶变换方法直接卷积https://zhuanlan.zhihu.com/p/624292239 Goodman, J. W. (2004). Intro…

Linux shell中的函数定义、传参和调用

Linux shell中的函数定义、传参和调用&#xff1a; 函数定义语法&#xff1a; [ function ] functionName [()] { } 示例&#xff1a; #!/bin/bash# get limit if [ $# -eq 1 ] && [ $1 -gt 0 ]; thenlimit$1echo -e "\nINFO: input limit is $limit" e…

智能优化算法应用:基于狮群算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于狮群算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于狮群算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.狮群算法4.实验参数设定5.算法结果6.参考文献7.MATLAB…

string的模拟

> 作者简介&#xff1a;დ旧言~&#xff0c;目前大二&#xff0c;现在学习Java&#xff0c;c&#xff0c;c&#xff0c;Python等 > 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 > 目标&#xff1a;能手撕模拟string类 > 毒鸡汤&#xff1a;时间…

强化学习(一)——基本概念及DQN

1 基本概念 智能体 agent &#xff0c;做动作的主体&#xff0c;&#xff08;大模型中的AI agent&#xff09; 环境 environment&#xff1a;与智能体交互的对象 状态 state &#xff1b;当前所处状态&#xff0c;如围棋棋局 动作 action&#xff1a;执行的动作&#xff0c;…

无脑018——win11部署whisper,语音转文字

1.conda创建环境 conda create -n whisper python3.9 conda activate whisper安装pytorch pip install torch1.8.1cu101 torchvision0.9.1cu101 torchaudio0.8.1 -f https://download.pytorch.org/whl/torch_stable.html安装whisper pip install -U openai-whisper2.准备模型…

代码随想录算法训练营第三十四天|62.不同路径,63. 不同路径 II

62. 不同路径 - 力扣&#xff08;LeetCode&#xff09; 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#…

分享几个可以免费使用GPT工具

1. 国产可以使用GPT3.5和4.0的网站&#xff0c;每日有免费的使用额度&#xff0c;响应速度&#xff0c;注册时不用使用手机号&#xff0c;等个人信息&#xff0c;注重用户隐私&#xff0c;好评&#xff01; 一个好用的ChatGPT系统 &#xff0c;可以免费使用3.5 和 4.0https://…

如何用Java实现扑克牌(附源码)

目录 一.扑克牌的数据结构 二.买牌(扑克牌的初始化) 三.洗牌 四.发牌 五.完整代码 Card.java CardList.java 六.测试 输出结果 一.扑克牌的数据结构 首先&#xff0c;扑克牌是一幅一幅的&#xff0c;除去大小王以外一共有52张&#xff0c;我们可以考虑用数组来存储…

Java高级技术-反射

认识反射、获取类 获取类的方法 获取类的构造器 获取类的构造器、并对其进行操作 获取构造器的作用&#xff1a;依然是初始化对象返回 获取成员变量 获取成员变量的方法 获取成员变量的作用&#xff1a;赋值、取值 获取类的成员方法 方法 作用&#xff1a;依然是执行 作用、…

Docker容器间网络共享

Docker容器间网络共享 1、新建网络2、容器绑定网卡3、验证 Docker环境中为了一套应用部署多个环境、并且不修改配置文件的情况下&#xff0c;做到一键部署。要求不同容器直接的网络交互&#xff0c;使用容器名称。 网络相关常用命令 #查看网络内部信息docker network inspect b…

Vim多行编辑

Vim多行编辑 Ctrlq进入多行编辑模式&#xff0c;然后上下选择要编辑的行 按下I或者Shifti&#xff0c;进入编辑模式 编辑的时候多行不会同时变化&#xff0c;不要担心&#xff0c;确实是多行编辑 编辑完成&#xff0c;想要结束多行编辑&#xff0c;按下Esc&#xff0c;此时…

前端小记--2.element-ui中级联选择器cascader如何默认展开下拉框

最近做项目时&#xff0c;遇到一个需求&#xff1a;在一个排班表中&#xff0c;展示人员的值班情况&#xff0c;点击单元格&#xff0c;弹出下拉框&#xff0c;修改人员排班信息。 由于下拉框选择内容是树状结构&#xff0c;这里使用了element-ui中级联组件cascader&#xff0c…

C-语言每日刷题

目录 [蓝桥杯 2015 省 A] 饮料换购 题目描述 输入格式 输出格式 输入输出样例 # [蓝桥杯 2023 省 A] 平方差 题目描述 输入格式 输出格式 输入输出样例 说明/提示 【样例说明】 [NOIP2001 普及组] 数的计算 题目描述 输入格式 输出格式 输入输出样例 说明/提示 样例 1 解释 数据…

TCP 重传、滑动窗口、流量控制、拥塞控制

1&#xff1a;重传机制 超时重传 快速重传 SACK 方法 Duplicate SACK 1&#xff1a;重传机制 超时重传&#xff1a;重传机制的其中一个方式&#xff0c;就是在发送数据时&#xff0c;设定一个定时器&#xff0c;当超过指定的时间后&#xff0c;没有收到对方的ACK确认应答报文…