算法沉淀——动态规划篇(子数组系列问题(下))

算法沉淀——动态规划篇(子数组系列问题(下))

  • 前言
  • 一、等差数列划分
  • 二、最长湍流子数组
  • 三、单词拆分
  • 四、环绕字符串中唯一的子字符串

前言

几乎所有的动态规划问题大致可分为以下5个步骤,后续所有问题分析都将基于此

  • 1.、状态表示:通常状态表示分为以下两种,其中更是第一种为主。

    • 以i为结尾,dp[i] 表示什么,通常为代求问题(具体依题目而定)
    • 以i为开始,dp[i]表示什么,通常为代求问题(具体依题目而定)
  • 2、状态转移方程

    • 以上述的dp[i]意义为根据, 通过最近一步来分析和划分问题,由此来得到一个有关dp[i]的状态转移方程。
  • 3、dp表创建,初始化

    • 动态规划问题中,如果直接使用状态转移方程通常会伴随着越界访问等风险,所以一般需要初始化。而初始化最重要的两个注意事项便是:保证后续结果正确,不受初始值影响;下标的映射关系
    • 初始化一般分为以下两种:
      • 直接初始化开头的几个值。
      • 一维空间大小+1,下标从1开始;二维增加一行/一列
  • 4、填dp表、填表顺序:根据状态转移方程来确定填表顺序。

  • 5、确定返回值

一、等差数列划分

【题目】:413. 等差数列划分
【题目】:

 如果一个数列 至少有三个元素 ,并且任意两个相邻元素之差相同,则称该数列为等差数列。例如,[1,3,5,7,9]、[7,7,7,7] 和 [3,-1,-5,-9] 都是等差数列。
 给你一个整数数组 nums ,返回数组 nums 中所有为等差数组的 子数组 个数。(子数组 是数组中的一个连续序列)

【示例】:

输入:nums = [1,2,3,4]
输出:3
解释:nums 中有三个子等差数组:[1, 2, 3]、[2, 3, 4] 和 [1,2,3,4] 自身。

【分析】:
 我们可以定义dp[i]表示以i为结尾,等差数组的子数组个数。之后我们可以通过判断(nums[i]、nums[i-1]、nums[i-2])是否构成等差数列,来进一步分析

状态转移方程推导:

  1. 如果nums[i]、nums[i-1]、nums[i-2]不构成等差数列,显然此时以i为结尾的等差数组的子数组个数为0。即dp[i] = 0;
  2. 如果构成等差数列,此时dp[i]的值至少为1。此时我们还需加上dp[i-1]的值。原因在于如果以i-1为结尾的等差数列存在,此时该等差数列公差为dp[i-1] -dp[i-2]。同时nums[i]、nums[i-1]、nums[i-2]构成等差数列,公差也为dp[i-1] -dp[i-2]。这也意味着,以i-1为结尾的所有等差数列,在添加新增nums[i]元素后,依然是等差数列。所以状态转移方程为dp[i] = dp[i - 1] + 1;

细节处理:
 显然当i为1、2时,状态转移方程不适用。我们由于dp[0]、dp[1]一定构不成等差数列,所以我们可以先将dp[0]、dp[1]先初始化为0,在从下标2开始,从左往右填表。

【代码编写】:

class Solution {
public:int numberOfArithmeticSlices(vector<int>& nums) {int n = nums.size();vector<int> dp(n);int ret = 0;for(int i = 2; i < n; i++){if(nums[i] - nums[i - 1] == nums[i - 1] - nums[i - 2])dp[i] = dp[i - 1] + 1;ret += dp[i];//累加所有结果}return ret;}
};

二、最长湍流子数组

【题目链接】:978. 最长湍流子数组
【题目】:

 给定一个整数数组 arr ,返回 arr 的 最大湍流子数组的长度 。如果比较符号在子数组中的每个相邻元素对之间翻转,则该子数组是 湍流子数组 。
 更正式地来说,当 arr 的子数组 A[i], A[i+1], …, A[j] 满足仅满足下列条件时,我们称其为湍流子数组:
若 i <= k < j :当 k 为奇数时, A[k] > A[k+1],且当 k 为偶数时,A[k] < A[k+1];
或 若 i <= k < j :当 k 为偶数时,A[k] > A[k+1] ,且当 k 为奇数时, A[k] < A[k+1]。

【示例】:

输入:arr = [9,4,2,10,7,8,8,1,9]
输出:5
解释:arr[1] > arr[2] < arr[3] > arr[4] < arr[5]

【分析】:
 我们定义f[i]表示以i位置为结尾,并且最后是“上升”趋势的最长湍流子数组大小;g[i]表示以i位置为结尾,并且最后是“下降”趋势的最长湍流子数组大小。

状态转移方程推导:
 此时以i为结尾的湍流子数组长度可能为1,或大于1。具体如下:

在这里插入图片描述

在这里插入图片描述
特殊处理:
 以i为结尾的湍流子数组中,不管最后一步是呈上升趋势还是下降趋势,最小长度一定为1,即nums[i]本身。所以我们在创建f和g表时,可以将初始值设为1。后续填表过程中,仅需考虑子数组长度大于1的情况即可!!

【代码编写】:

class Solution {
public:int maxTurbulenceSize(vector<int>& arr) {int n = arr.size();vector<int> f(n, 1), g(n, 1);int ret = 1;for(int i = 1; i < n; i++){if(arr[i] > arr[i - 1])f[i] = g[i - 1] + 1;else if(arr[i] < arr[i - 1])g[i] = f[i - 1] + 1;ret = max(ret, max(f[i], g[i]));}return ret;}
};

三、单词拆分

【题目链接】:139. 单词拆分
【题目】:

 给你一个字符串 s 和一个字符串列表 wordDict 作为字典。如果可以利用字典中出现的一个或多个单词拼接出 s 则返回 true。
 注意:不要求字典中出现的单词全部都使用,并且字典中的单词可以重复使用。

【示例】:

输入: s = “applepenapple”, wordDict = [“apple”, “pen”]
输出: true
解释: 返回 true 因为 “applepenapple” 可以由 “apple” “pen” “apple” 拼接成。
注意,你可以重复使用字典中的单词。

【分析】:
 我们可以定义dp[i]表示以i位置为结尾的子字符串是否能单词拆分。

状态转移方程推导:

 要判断从下标从0到i的子串是否能单词拆分。我们可以将0到i的字串分为两部分:0到j-1,j到i(0 <= j <= i)。而dp[j-1]表示以j-1位置为结尾的字串能否单词拆分的结果。此时我们还需判断下标从j到i的字串是否能单词拆分。此时即可判断此种分发是否能实现单词拆分!!
 但由于j的位置不确定。所以我们可以一次将j从开始,逐渐减小到起始下标0。在每次递减过程中,只有存在一种拆分发将拆分出的两个字串都能实现拆分单词,此时dp[i]=true,同时可停止遍历。否则为false;

在这里插入图片描述
细节处理:
 在填dp表过程中,dp[i]的值会用到dp[j-1](0<=j<=i),可能会发生越界访问。所以我们为dp表额外增加一个空间,同时为了保证后续填表的正确性,我们需要将dp[0]初始化为true。
 同时面对字符串问题时,通常需要存在子字符窜问题。此时,下标映射关系可能+1,可能减1。所以这里个原始字符串最开始任意增加一个字符(习惯上该字符为空字符),让原始字符串下标统一向后移动一位。
【代码编写】:

class Solution {
public:bool wordBreak(string s, vector<string>& wordDict) {unordered_map<string, int> hash;for(auto& str : wordDict)//后续快速查找是否存在某单词hash[str]++;int n = s.size();vector<bool> dp(n + 1);dp[0] = true;//初始化,保证后续填表正确s = ' ' + s;//让s下标集体向后移动一位for(int i = 1; i <= n; i++)for(int j = i; j >= 1; j--){if(dp[j - 1] && hash.count(s.substr(j, i - j + 1))){dp[i] = true;break;}}return dp[n];}
};

四、环绕字符串中唯一的子字符串

【题目链接】:467. 环绕字符串中唯一的子字符串
【题目】:

【代码编写】:

 定义字符串 base 为一个 “abcdefghijklmnopqrstuvwxyz” 无限环绕的字符串,所以 base 看起来是这样的:

  • “…zabcdefghijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyzabcd…”。
    给你一个字符串 s ,请你统计并返回 s 中有多少 不同非空子串 也在 base 中出现。

【示例】:

输入:s = “zab”
输出:6
解释:字符串 s 有六个子字符串 (“z”, “a”, “b”, “za”, “ab”, and “zab”) 在 base 中出现。

【分析】:
 我们可以定义dp[i]表示以i位置为结尾的字符串中非空字串在base中出现的个数。

状态转移方程推导:
 非空字串存在于base中有两种可能:

  1. 相邻字符是连续的,即s[i-1] + 1 == s[i]
  2. 相邻字符分别是26个小写字母的结束和开始,即是bashs[i-1] == 'z' && s[i] == 'a'
    所以状态转移方程为:
if((s[i] - s[i - 1] == 1) || (s[i - 1] == 'z' && s[i] == 'a'))dp[i] = dp[i - 1] + 1;

细节处理:
 由于当个字符一定存在于base中,所以dp[i]的值最小为1,所以我们可以将dp表中的初始值全部初始化为1。

 上述dp表中的结果存在重复值,不能直接累加。那如何去重?

  • 我们知道以某一个字符为结尾的子串中,长子串一定包含了短子串的所有结果。所以我们可以借助一个26空间大小的数组,将s分割出的字串中,以结尾字符为依据,将最长字串结果放入对应的数组空间中。从而实现去重效果。即:hash[s[i] - 'a'] = max(hash[s[i] - 'a'], dp[i]);
     既然以及去重了,最后只需将数组中的结果累加即可!!


【代码编写】:

class Solution {
public:int findSubstringInWraproundString(string s) {int n = s.size();vector<int> dp(n, 1);for(int i = 1; i < n; i++)if((s[i] - s[i - 1] == 1) || (s[i - 1] == 'z' && s[i] == 'a'))dp[i] = dp[i - 1] + 1;int hash[26] = {0};for(int i = 0; i < n; i++)//去重hash[s[i] - 'a'] = max(hash[s[i] - 'a'], dp[i]);int ret = 0;for(auto x : hash)ret += x;return ret;}
};

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

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

相关文章

Jenkins首次安装选择推荐插件时出现”No such plugin cloudbees-folder”解决方案

安装Jenkins成功之后&#xff0c;首次启动Jenkins后台管理&#xff0c;进入到安装插件的步骤&#xff0c;选择"推荐安装"&#xff0c;继续下一步的时候出现错误提示&#xff1a; 出现一个错误 安装过程中出现一个错误&#xff1a;No such plugin&#xff1a;cloudb…

Linux的开发工具(二):编译器gcc/g++与Linux项目自动化构建工具-Makefile

目录 Linux的编译器-gcc/g 问题一&#xff1a;gcc有时候为什么不能编译带有for循环的c语言源文件&#xff1f; 问题二&#xff1a;gcc中c源文件的后缀是什么&#xff1f; 问题三&#xff1a;gcc能编译c的源文件文件吗&#xff1f; 问题四&#xff1a;如何查看自己是否拥有…

jmeter链路压测

比如登录后返回token&#xff0c;业务打印上传的操作需要用到token 线程组中添加登录请求&#xff0c;并执行 1、添加登录并执行&#xff0c;查看结果 2、结果树中下拉选择正则表达式&#xff0c;将token参数和值复制粘贴到下方&#xff0c;将token值改为(.*?)&#xff0…

关于第十二届蓝桥杯时间显示题中包和模块的使用解释

题目信息&#xff1a; 解题代码&#xff1a; from datetime import datetime, timedelta # 定义起始时间&#xff0c;即 Unix 时间戳的零点&#xff08;1970年1月1日&#xff09; start datetime(year1970, month1, day1) # 定义时间间隔&#xff0c;这里以毫秒为单位 dela …

js类型转换

类型转换只有这四种&#xff0c;例如如果要对象转数字&#xff0c;那么就需要先把对象转成原始类型&#xff0c;再从原始类型转到数字。 空数组转原始类型是一个空字符串。空对象转原始类型是[object Object]。 let a {} console.log(a);// NaN //等价于 a->原始 然后原始…

适用于 Linux 的 Windows 子系统安装初体验

1、简述 Windows Subsystem for Linux (WSL) 是 Windows 的一项功能&#xff0c;允许您在 Windows 计算机上运行 Linux 环境&#xff0c;而无需单独的虚拟机或双重启动。 WSL 旨在为想要同时使用 Windows 和 Linux 的开发人员提供无缝且高效的体验。 使用 WSL 安装和运行各种 L…

InternLM2-lesson2作业

书生浦语大模型趣味 Demo 视频连接&#xff1a;https://www.bilibili.com/video/BV1AH4y1H78d/?vd_source902e3124d4683c41b103f1d1322401fa 目录 书生浦语大模型趣味 Demo一、基础作业二、进阶作业 一、基础作业 第一次执行&#xff1a; 第二次执行&#xff1a; 第一次执…

四核8g服务器价格多少钱?

2024年腾讯云4核8G服务器租用优惠价格&#xff1a;轻量应用服务器4核8G12M带宽646元15个月&#xff0c;CVM云服务器S5实例优惠价格1437.24元买一年送3个月&#xff0c;腾讯云4核8G服务器活动页面 txybk.com/go/txy 活动链接打开如下图&#xff1a; 腾讯云4核8G服务器优惠价格 轻…

Doris实践——同程数科实时数仓建设

目录 前言 一、早期架构演进 二、Doris和Clickhouse选型对比 三、新一代统一实时数据仓库 四、基于Doris的一站式数据平台 4.1 一键生成任务脚本提升任务开发效率 4.2 自动调度监控保障任务正常运行 4.3 安全便捷的可视化查询分析 4.4 完备智能的集群监控 五、收益与…

基于单片机的无线红外报警系统

**单片机设计介绍&#xff0c;基于单片机的无线红外报警系统 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机的无线红外报警系统是一种结合了单片机控制技术和无线红外传感技术的安防系统。该系统通过无线红外传感器实…

Excel、PowerQuery 和 ChatGPT 终极手册(下)

原文&#xff1a;Ultimate ChatGPT Handbook for Enterprises 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 使用 SUMIFS、SUMPRODUCT、AGGREGATE 和 MAX 函数查找数值数据 其中之一鲜为人知的事实是&#xff0c;当查找单个数值时&#xff0c;匹配和三角函数可能比查…

哈佛大学商业评论 -- 第二篇:增强现实是如何工作的?

AR将全面融入公司发展战略&#xff01; AR将成为人类和机器之间的新接口&#xff01; AR将成为人类的关键技术之一&#xff01; 请将此文转发给您的老板&#xff01; --- 本文作者&#xff1a;Michael E.Porter和James E.Heppelmann 虽然物理世界是三维的&#xff0c;但大…

视觉大模型--deter的深入理解

但对于transformer用于目标检测领域的开创性模型&#xff0c;该模型言简意赅&#xff0c;但是但从论文理解&#xff0c;有很多细节都不清楚&#xff0c;尤其是解码器的query和二分图匹配(Bipartite Matching)和匈牙利算法(Hungarian Algorithm)相关&#xff0c;本文将根据代码详…

Android自定义view;实现掌阅打开书籍动画效果

这里利用自定义view的方式来处理&#xff0c;初始化数据&#xff0c;camera通过setLocation调整相机的位置&#xff0c;但是Camera 的位置单位是英寸&#xff0c;英寸和像素的换算单位在 Skia 中被写成了72 像素&#xff0c;8 x 72 576&#xff0c;所以它的默认位置是 (0, 0, …

文件操作(详解)

该片博客有点长大家可以通过目录选择性阅读 这是个人主页 敲上瘾-CSDN博客 目录 1. 为什么使⽤⽂件&#xff1f; 2. 什么是⽂件&#xff1f; 2.1 程序⽂件 2.2 数据⽂件 2.3 ⽂件名 3. ⼆进制⽂件和⽂本⽂件&#xff1f; 4. ⽂件的打开和关闭 4.1 流和标准流 4.1.1 流…

Java 包装类初识泛型

登神长阶 第六阶 包装类&初识泛型 目录 &#x1f600;一.包装类 &#x1f604;1.基本数据类型以及其对应的包装类 &#x1f602;2.装箱和拆箱 &#x1f607;2.1.装箱&#xff08;Boxing&#xff09; &#x1f609;2.2.拆箱&#xff08;Unboxing&#xff09; &#x…

【项目技术介绍篇】若依项目代码文件结构介绍

作者介绍&#xff1a;本人笔名姑苏老陈&#xff0c;从事JAVA开发工作十多年了&#xff0c;带过大学刚毕业的实习生&#xff0c;也带过技术团队。最近有个朋友的表弟&#xff0c;马上要大学毕业了&#xff0c;想从事JAVA开发工作&#xff0c;但不知道从何处入手。于是&#xff0…

55 npm run serve 和 npm run build 的分包策略

前言 这里我们来看一下 vue 这边 打包的时候的一些 拆分包的一些策略 我们经常会使用到 npm run build 进行服务的打包 然后 打包出来的情况, 可能如下, 可以看到 chunk-vendors 是进行了包的拆分, 我们这里就是 来看一下 这里 npm run build 的时候的, 一个分包的策略 测试…

【Linux实验室】NFS、DHCP的搭建

NFS、DHCP的搭建 1、nfs服务搭建及测试什么是NFS&#xff1f;环境准备服务端机器安装nfs-utils和rpcbind包启动NFS服务创建/data/NFSdata目录&#xff0c;配置nfs文件启动服务挂载测试在服务端在共享目录下创建文件测试在客户端在共享目录下创建文件 2、dhcp服务搭建及测试什么…

如何保护IP地址不被泄露?

当互联网成为每个家庭的重要组成部分后&#xff0c;IP地址就成了你的虚拟地址。您的请求从该地址开始&#xff0c;然后 Internet 将消息发送回该地址。那么&#xff0c;您担心您的地址被泄露吗&#xff1f; 对于安全意识高或者某些业务需求的用户&#xff0c;如果您正在寻找保护…