当前位置: 首页 > news >正文

dp练习【4】

最长数对链 

646. 最长数对链

给你一个由 n 个数对组成的数对数组 pairs ,其中 pairs[i] = [lefti, righti] 且 lefti < righti 。

现在,我们定义一种 跟随 关系,当且仅当 b < c 时,数对 p2 = [c, d] 才可以跟在 p1 = [a, b] 后面。我们用这种形式来构造 数对链 。

找出并返回能够形成的 最长数对链的长度 。

你不需要用到所有的数对,你可以以任何顺序选择其中的一些数对来构造。

示例 1:

输入:pairs = [[1,2], [2,3], [3,4]]
输出:2
解释:最长的数对链是 [1,2] -> [3,4] 。

示例 2:

输入:pairs = [[1,2],[7,8],[4,5]]
输出:3
解释:最长的数对链是 [1,2] -> [4,5] -> [7,8] 。
class Solution {public int findLongestChain(int[][] pairs) {int n = pairs.length;Arrays.sort(pairs, (a, b) -> a[0] - b[0]);int[] dp = new int[n];Arrays.fill(dp, 1);for (int i = 0; i < n; i++) {for (int j = 0; j < i; j++) {if (pairs[i][0] > pairs[j][1]) {// 通过i位置与前面所有的比较,然后得到当前时候最多有多少对(j到i之间最大对数)dp[i] = Math.max(dp[i], dp[j] + 1);}}}return dp[n - 1];}
}

最长定差子序列

1218. 最长定差子序列

给你一个整数数组 arr 和一个整数 difference,请你找出并返回 arr 中最长等差子序列的长度,该子序列中相邻元素之间的差等于 difference 。

子序列 是指在不改变其余元素顺序的情况下,通过删除一些元素或不删除任何元素而从 arr 派生出来的序列。

示例 1:

输入:arr = [1,2,3,4], difference = 1
输出:4
解释:最长的等差子序列是 [1,2,3,4]。

示例 2:

输入:arr = [1,3,5,7], difference = 1
输出:1
解释:最长的等差子序列是任意单个元素。

示例 3:

输入:arr = [1,5,7,8,5,3,4,2,1], difference = -2
输出:4
解释:最长的等差子序列是 [7,5,3,1]。

这个题目不能用常规dp来做,会超时,比如说这样,就超时了

class Solution {public int longestSubsequence(int[] arr, int difference) {int n = arr.length;int[] dp = new int[n + 1];Arrays.fill(dp, 1);int maxLen = 1;for (int i = 1; i < n; i++) {for (int j = 0; j < i; j++) {if (arr[i] - arr[j] == difference) {dp[i] = Math.max(dp[i], dp[j] + 1);}}maxLen = Math.max(maxLen,dp[i]);}return maxLen;}
}

 

要用哈希表来存储之前最长的重复子数组才行

class Solution {// 方法用于计算数组 arr 中具有固定差值 difference 的最长子序列长度public int longestSubsequence(int[] arr, int difference) {int ans = 0; // 初始化最长子序列的长度为 0// 创建一个大小为 40001 的整型数组 dp,用于存储动态规划的状态// 数组大小的选择是为了适应输入数据的变化范围,假设输入数据范围在 [-20000, 19999] 内int[] dp = new int[40001];// 遍历数组 arr 的每一个元素for (int num : arr) {// 为了使数组下标非负,将当前数字 num 偏移 20000 后使用作为 dp 数组的索引// dp[num + 20000] 存储了以 num 结尾的具有固定差值 difference 的最长子序列长度// dp[num + 20000 - difference] 则是前一个元素 (num - difference) 的最长子序列长度// 当前元素 num 的最长子序列长度为前一个元素的长度加 1dp[num + 20000] = dp[num + 20000 - difference] + 1;// 更新全局最长子序列长度 ans// 取当前已知的最长子序列长度和 dp[num + 20000] 中较大的值ans = Math.max(ans, dp[num + 20000]);}// 返回最长子序列的长度return ans;}
}

最长等差数列

1027. 最长等差数列

给你一个整数数组 nums,返回 nums 中最长等差子序列的长度

回想一下,nums 的子序列是一个列表 nums[i1], nums[i2], ..., nums[ik] ,且 0 <= i1 < i2 < ... < ik <= nums.length - 1。并且如果 seq[i+1] - seq[i]0 <= i < seq.length - 1) 的值都相同,那么序列 seq 是等差的。

示例 1:

输入:nums = [3,6,9,12]
输出:4
解释: 
整个数组是公差为 3 的等差数列。

示例 2:

输入:nums = [9,4,7,2,10]
输出:3
解释:
最长的等差子序列是 [4,7,10]。

示例 3:

输入:nums = [20,1,15,3,10,5,8]
输出:4
解释:
最长的等差子序列是 [20,15,10,5]。
class Solution {// 方法接收一个整数数组 nums 作为参数public int longestArithSeqLength(int[] nums) {int n = nums.length; // 获取数组长度// 创建一个二维数组 dp,大小为 [n][1001],用于存储动态规划的结果int[][] dp = new int[n][1001];int maxLen = 0; // 初始化最长等差数列的长度为0// 从第二个元素开始遍历数组for (int k = 1; k < n; k++) {// 从第一个元素到第k-1个元素遍历for (int j = 0; j < k; j++) {// 计算两数之差,并加上500以保证下标非负// 这里假设差值的范围是 [-500, 500],因此差值加上500后可以作为下标int d = nums[k] - nums[j] + 500;// 根据差值 d 更新 dp[k][d] 的值,即以 nums[k] 结尾且差值为 d 的等差数列的长度// dp[j][d] 表示以 nums[j] 结尾且差值为 d 的等差数列的长度// 加1是因为当前元素 nums[k] 可以与前面的等差数列形成新的等差数列dp[k][d] = dp[j][d] + 1;// 更新最长等差数列的长度maxLen = Math.max(maxLen, dp[k][d]);}}// 最终返回最长等差数列的长度,由于 dp 数组中存储的是除第一个元素外的等差数列长度,所以需要加1return maxLen + 1;}
}

腾讯面试时的算法题目

牛客上看面经,看到有人写的,在面试之后,面试官出的两道dp。自己尝试了一下,发现难度还好,属于动态规划系列中【最长递增子序列】分类和【动态规划在字符串的应用】分类中类似的题型

最长重复子数组

718. 最长重复子数组

给两个整数数组 nums1 和 nums2 ,返回 两个数组中 公共的 、长度最长的子数组的长度 

示例 1:

输入:nums1 = [1,2,3,2,1], nums2 = [3,2,1,4,7]
输出:3
解释:长度最长的公共子数组是 [3,2,1] 。

示例 2:

输入:nums1 = [0,0,0,0,0], nums2 = [0,0,0,0,0]
输出:5

提示:

  • 1 <= nums1.length, nums2.length <= 1000
  • 0 <= nums1[i], nums2[i] <= 100
class Solution {public int longestArithSeqLength(int[] nums) {int n=nums.length;int[][] dp=new int[n][1001];int maxLen=0;//保存结果for(int k=1;k<n;k++){for(int j=0;j<k;j++){int d=nums[k]-nums[j]+500;//统一加偏移量,使下标非负dp[k][d]=dp[j][d]+1; //根据 d 去填充dp[k][d]maxLen=Math.max(maxLen,dp[k][d]);//维护最大值}}return maxLen+1;}
}

最长公共子序列

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 
class Solution {public int longestCommonSubsequence(String text1, String text2) {int m = text1.length();int n = text2.length();int[][] dp = new int[n + 1][m + 1];for (int i = 1; i <= n; i++) {for (int j = 1; j <= m; j++) {if (text1.charAt(j - 1) == text2.charAt(i - 1)) {dp[i][j] = dp[i - 1][j - 1] + 1;} else {dp[i][j] = Math.max(dp[i][j - 1], dp[i - 1][j]);}}}return dp[n][m];}
}

 

每日一题

3174. 清除数字

给你一个字符串 s 。

你的任务是重复以下操作删除 所有 数字字符:

  • 删除 第一个数字字符 以及它左边 最近 的 非数字 字符。

请你返回删除所有数字字符以后剩下的字符串。

示例 1:

输入:s = "abc"

输出:"abc"

解释:

字符串中没有数字。

示例 2:

输入:s = "cb34"

输出:""

解释:

一开始,我们对 s[2] 执行操作,s 变为 "c4" 。

然后对 s[1] 执行操作,s 变为 "" 。

用栈的思想解题:

class Solution {public String clearDigits(String s) {StringBuilder res = new StringBuilder();for (char c : s.toCharArray()){if (Character.isDigit(c)) {res.deleteCharAt(res.length() - 1);} else {res.append(c);}}return res.toString();}
}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • C语言刷题日记(附详解)(4)
  • 【机器学习-神经网络】卷积神经网络
  • Hadoop常用命令
  • 2022 年高教社杯全国大学生数学建模竞赛-C 题 古代玻璃制品的成分分析与鉴别详解+分类模型Python代码源码
  • 【知识点】图论续篇 - 最短路算法合集
  • A02、Java编程性能调优(02)
  • 论文速读|重新审视奖励设计与评估:用于强健人型机器人站立与行走控制的方法
  • 设计模式学习-简单的命令模式例子
  • AI科学家:自动化科研的未来之路
  • S3C2440开发板:时钟,PWM定时器控制蜂鸣器发声
  • 小白遇上字符串解析问题,正则和原生字符串函数谁来救场?
  • 双绞线如何抑制传导干扰
  • DigitalOcean Kubernetes引入NVIDIA H100 GPU,助力 AI/ML 创新
  • 第R2周:LSTM-火灾温度预测
  • Java使用POI创建带样式和公式的Excel文件
  • 《剑指offer》分解让复杂问题更简单
  • 「面试题」如何实现一个圣杯布局?
  • Date型的使用
  • electron原来这么简单----打包你的react、VUE桌面应用程序
  • Golang-长连接-状态推送
  • iOS动画编程-View动画[ 1 ] 基础View动画
  • LeetCode刷题——29. Divide Two Integers(Part 1靠自己)
  • Linux各目录及每个目录的详细介绍
  • mysql外键的使用
  • node学习系列之简单文件上传
  • SQLServer之索引简介
  • Vue源码解析(二)Vue的双向绑定讲解及实现
  • 从setTimeout-setInterval看JS线程
  • 给自己的博客网站加上酷炫的初音未来音乐游戏?
  • 后端_MYSQL
  • 排序算法学习笔记
  • 浅谈Kotlin实战篇之自定义View图片圆角简单应用(一)
  • 如何设计一个比特币钱包服务
  • 通过git安装npm私有模块
  • 我的面试准备过程--容器(更新中)
  • 协程
  • 责任链模式的两种实现
  • 这几个编码小技巧将令你 PHP 代码更加简洁
  • 3月27日云栖精选夜读 | 从 “城市大脑”实践,瞭望未来城市源起 ...
  • 教程:使用iPhone相机和openCV来完成3D重建(第一部分) ...
  • ​埃文科技受邀出席2024 “数据要素×”生态大会​
  • ​香农与信息论三大定律
  • ​一文看懂数据清洗:缺失值、异常值和重复值的处理
  • # 计算机视觉入门
  • #Datawhale AI夏令营第4期#AIGC文生图方向复盘
  • #基础#使用Jupyter进行Notebook的转换 .ipynb文件导出为.md文件
  • ${factoryList }后面有空格不影响
  • $NOIp2018$劝退记
  • (¥1011)-(一千零一拾一元整)输出
  • (1)Hilt的基本概念和使用
  • (AngularJS)Angular 控制器之间通信初探
  • (leetcode学习)236. 二叉树的最近公共祖先
  • (Redis使用系列) Springboot 实现Redis消息的订阅与分布 四
  • (二)斐波那契Fabonacci函数
  • (六)Hibernate的二级缓存