每日OJ题_子序列dp⑧_力扣446. 等差数列划分 II - 子序列

目录

力扣446. 等差数列划分 II - 子序列

解析代码


力扣446. 等差数列划分 II - 子序列

446. 等差数列划分 II - 子序列

难度 困难

给你一个整数数组 nums ,返回 nums 中所有 等差子序列 的数目。

如果一个序列中 至少有三个元素 ,并且任意两个相邻元素之差相同,则称该序列为等差序列。

  • 例如,[1, 3, 5, 7, 9][7, 7, 7, 7] 和 [3, -1, -5, -9] 都是等差序列。
  • 再例如,[1, 1, 2, 5, 7] 不是等差序列。

数组中的子序列是从数组中删除一些元素(也可能不删除)得到的一个序列。

  • 例如,[2,5,10] 是 [1,2,1,2,4,1,5,10] 的一个子序列。

题目数据保证答案是一个 32-bit 整数。

示例 1:

输入:nums = [2,4,6,8,10]
输出:7
解释:所有的等差子序列为:
[2,4,6]
[4,6,8]
[6,8,10]
[2,4,6,8]
[4,6,8,10]
[2,4,6,8,10]
[2,6,10]

示例 2:

输入:nums = [7,7,7,7,7]
输出:16
解释:数组中的任意子序列都是等差子序列。

提示:

  • 1  <= nums.length <= 1000
  • -2^31 <= nums[i] <= 2^31 - 1
class Solution {
public:int numberOfArithmeticSlices(vector<int>& nums) {}
};

解析代码

力扣873. 最长的斐波那契子序列的长度、力扣1027. 最长等差数列类似,动态规划解法思路:

状态表示:以某个位置为结尾,结合题目要求,先定义一个状态表示:

dp[i] 表示:以 i 位置元素为结尾的所有子序列中,等差数列的个数

        但是这里有⼀个非常致命的问题,那就是我们无法确定 i 结尾的斐波那契序列的样子。这样就会导致我们无法推导状态转移方程,因此我们定义的状态表示需要能够确定一个等差数列

        根据等差数列的特性,我们仅需知道序列里面的最后两个元素,就可以确定这个序列的样子。因此,修改状态表示为:

dp[i][j] 表示:以 i 位置以及 j 位置的元素为结尾的所有的子序列中,等差数列的个数。规定一下 i < j 。

状态转移方程:

设 nums[i] = b, nums[j] = c ,那么这个序列的前一个元素就是 a = 2 * b - c。根据 a 的情况讨论:

  • a 存在,下标为 k ,并且 a < b :此时我们需要以 k 位置以及 i 位置元素为结尾的等差数列的长度,然后再加上 j 位置的元素(+1)即可。于是 dp[i][j] =dp[k][i] + 1 ; 因为 a 可能有很多个,我们需要全部累加起来。p[i][j] +=dp[k][i] + 1 ;
  • a 存在,但是 b < a < c :dp[i][j] =0 ;
  • a 不存在: dp[i][j] = 0 ;

综上,状态转移方程分情况讨论即可。

        优化点:我们发现,在状态转移方程中,我们需要确定 a 元素的下标。因此我们可以在 dp 之前,将所有的元素和下标数组绑定在⼀起,放到哈希表中。这里为什么保存下标数组,是因为要统计个数,所有的下标都需要统计。

        初始化:可以将表里面的值都初始化为 0 。

        填表顺序:先固定斐波那契子序列的最后一个数,然后枚举倒数第二个数。

        返回值:返回 dp 表中的所有值的和。

class Solution {
public:int numberOfArithmeticSlices(vector<int>& nums) {int n = nums.size(), ret = 0;vector<vector<int>> dp(n, vector<int>(n, 0));// dp[i][j] 表示:以 i 位置以及 j 位置的元素为结尾的所有的子序列中,等差数列的个数。i < j unordered_map<long long, vector<int>> hash(n);for(int i = 0; i < n; ++i){hash[nums[i]].push_back(i);}for(int j  = 2; j < n; ++j) // 固定倒数第一个数{for(int i = 1; i < j; ++i) // 先固定倒数第二个数{long long a = (long long)2 * nums[i] - nums[j]; // 防溢出if(hash.count(a)){for(auto& k : hash[a]){if(k < i)dp[i][j] += dp[k][i] + 1;elsebreak;}}ret += dp[i][j];}}return ret;}
};

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

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

相关文章

脑部肿瘤检测YOLOV8

脑部肿瘤检测&#xff0c;采用YOLOV8训练得到PT模型&#xff0c;然后转换成ONNX&#xff0c;OPENCV调用&#xff0c;支持C/PYTHON/ANDORID开发脑部肿瘤检测YOLOV8

如何在Plesk面板备份网站

本周有一个客户&#xff0c;购买Hostease的Windows虚拟主机&#xff0c;咨询我们的在线客服&#xff0c;询问Windows虚拟主机Plesk面板是否提供备份功能。我们为用户提供教程&#xff0c;用户很快完成了数据备份。在此&#xff0c;我们分享这个操作教程&#xff0c;希望可以对您…

FastAPI+React全栈开发19 React Hooks事件和状态

Chapter04 Setting Up a React Workflow 19 React Hooks envents and state FastAPIReact全栈开发19 React Hooks事件和状态 A great definition of React or its components is that it is, essentially, a function that converts a state to a user interface, a React c…

掌握Python的多方式分支——switch case 实现详解

&#x1f340; 前言 博客地址&#xff1a; CSDN&#xff1a;https://blog.csdn.net/powerbiubiu &#x1f44b; 简介 在 Python 编程语言中&#xff0c;没有内置的 switch case 功能。switch case 是一种常见的编程结构&#xff0c;它可以根据不同的条件值执行不同的代码块。…

机器学习之局部异常因子算法(Local Outlier Factor)

概念 局部异常因子算法(Local Outlier Factor, LOF)是一种用于检测数据集中的异常点的算法。它是一种无监督学习方法,不需要先验标签来识别异常值。相反,它利用数据点周围的局部邻域信息来计算每个数据点的异常程度。 LOF算法的核心思想是,异常点通常在其周围的邻域中具…

实践笔记-harbor-01搭建(版本:2.9.0)

harbor搭建 1.下载安装包&#xff08;版本&#xff1a;2.9.0&#xff09;2.修改配置文件3.安装4.访问harbor5.可能用得上的命令: 环境&#xff1a;centos7 1.下载安装包&#xff08;版本&#xff1a;2.9.0&#xff09; 网盘资源&#xff1a;https://pan.baidu.com/s/1fcoJIa4x…

多微信聚合聊天神器,让你的社交更高效!

对于那些拥有多个微信号的用户来说&#xff0c;频繁地在不同微信号和设备之间切换既麻烦又容易搞混。这时候&#xff0c;一款多微信聚合聊天神器——微信管理系统应运而生&#xff0c;为我们带来了极大的便利与高效。 下面一起来看看它都有哪些功能吧&#xff01; 1、多微信同…

mybatis plus 的方法有些带填充和逻辑删除,有些又不带

有没有人知道这是怎么个规律。逻辑删除看了下都会自动带上&#xff0c;但是主要是删除和修改方法&#xff0c;有些会按照设置的handler自动填充&#xff0c;有些又不会。有大佬知道吗。

C++ | Leetcode C++题解之第2题两数相加

题目&#xff1a; 题解&#xff1a; class Solution { public:vector<int> twoSum(vector<int>& nums, int target) {map<int,int> a;//提供一对一的hashvector<int> b(2,-1);//用来承载结果&#xff0c;初始化一个大小为2&#xff0c;值为-1的容…

软文推广4大坑,媒介盒子告诉你

今天媒介盒子来和大家聊聊软文推广中的4个坑&#xff0c;让企业在软文推广过程中少走弯路。 一、 没有目的 在如今这个“内容为王”的时代&#xff0c;对软文推广而言&#xff0c;不管其目的性是增强用户参与度、提升品牌认知度还是促成转化等&#xff0c;但总归一个原则&…

Linux网卡bond的七种模式详解

像Samba、Nfs这种共享文件系统&#xff0c;网络的吞吐量非常大&#xff0c;就造成网卡的压力很大&#xff0c;网卡bond是通过把多个物理网卡绑定为一个逻辑网卡&#xff0c;实现本地网卡的冗余&#xff0c;带宽扩容和负载均衡&#xff0c;具体的功能取决于采用的哪种模式。 Lin…

AI工作站设计方案:903-多路PCIe3.0的单CPU 学习型AI工作站

多路PCIe3.0的单CPU 学习型AI工作站 一、机箱功能和技术指标&#xff1a; 系统 系统型号 ORI-SR500 主板支持 EEB(12*13)/CEB(12*10.5)/ATX(12*9.6)/Mi cro ATX 前置硬盘 最大支持2个3.5寸1个2.5寸SATA 硬盘2个2.5寸SATA 硬盘 &#xff08;背部&#xff09; 电源类型 C…

Pytorch:torchvision.transforms.Compose

transforms.Compose 是PyTorch库中torchvision.transforms模块提供的一个功能&#xff0c;它允许将多个图像变换操作组合起来。当你在处理图像&#xff0c;并需要依次应用多个变换&#xff08;如缩放、裁剪、归一化等&#xff09;时&#xff0c;Compose可以把这些变换串联成一个…

C++教学——从入门到精通 8.bool逻辑变量

这次我们来学一个新的类型——bool 你知道么&#xff0c;bool他只有两个值&#xff0c;分别是true和false&#xff0c;意思分别是正确和错误&#xff0c;赋值定义&#xff1a;bool ntrue/false; 如果他原本的值是true&#xff0c;那么想要把他反过来就可以用n!n; 来编个简单…

探索父进程和子进程

文章目录 通过系统调用查看进程PID父进程、子进程 通过系统调用创建进程-fork初识为什么fork给父进程返回子进程的PID&#xff0c;给子进程返回0fork函数如何做到返回两个值一个变量为什么同时会有两个返回值&#xff1f;bash总结 通过系统调用查看进程PID getpid()函数可以获…

Java设计之道:色即是空,空即是色

引子 我们的这个世界上&#xff0c;存在这么一种东西&#xff1a; 第一&#xff1a;它不占据任何3D之体积&#xff0c;即它没有Volume第二&#xff1a;它也不占据任何2D之面积&#xff0c;即它没有Area第三&#xff1a;它也不占据任何1D之长度&#xff0c;即它没有Length 总之…

CentOS7 RPM升级支持BBR TCP/CC的内核版本

列出安装的内核 rpm -qa kernel # yum list installed kernel 删除已安装内核 sudo dnf remove kernel-4.0.4-301.fc22.x86_64 安装内核 rpm --import https://www.elrepo.org/RPM-GPG-KEY-elrepo.org rpm -Uvh http://www.elrepo.org/elrepo-release-7.0-2.el7.elrepo.noar…

【二叉树】Leetcode 101. 对称二叉树【简单】

对称二叉树 给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 示例1&#xff1a; 输入&#xff1a;root [1,2,2,3,4,4,3] 输出&#xff1a;true 解题思路 判断一棵二叉树是否是轴对称的&#xff0c;可以通过递归方式进行判断。 1、定义一个递归函数isMirr…

SpringSecurity学习总结(三更草堂)

SpringSecurity安全框架的核心功能是认证和授权&#xff1a; 认证&#xff1a;验证当前访问系统的是不是本系统的用户&#xff0c;并且要确认具体是哪个用户。 授权&#xff1a;经过认证后判断当前用户是否具有进行某个操作的权限。 一般来说中大型的项目都是使用SpringSecurit…

vue3组合式函数

vue3的组合式函数的作用是封装和复用响应式状态的函数。只能在setup 标签的script标签汇总或者setup函数中使用。 普通的函数只能调用一次&#xff0c;但是组合式函数接受到响应式参数&#xff0c;当该值发生变化时&#xff0c;也会触发相关函数的重新加载。 如下 定义了一个…