统计上升四元组

news/2024/9/23 10:17:16/

🌈个人主页:Yui_
🌈Linux专栏:Linux
🌈C语言笔记专栏:C语言笔记
🌈数据结构专栏:数据结构
🌈C++专栏:C++

文章目录

  • 1. 题目描述
  • 2. 解释
  • 3. DP前缀和+枚举

1. 题目描述

给你一个长度为n下标从0开始的整数数组nums,它包含1n的所有数字,请你返回上升四元组的数目。
如果一个四元组(i,j,k,l)满足以下条件,我们称其为上升的:

  • 0 <= i < j < k < l < n
  • nums[i] < nums[k] < nums[j] < nums[l]

示例1
输入:nums = [1,3,2,4,5]
输出:2
解释:

  1. 当i = 0,j = 1。k = 2 且 l = 3时,有nums[i] < nums[k] < nums[j] < nums[l]。
  2. 当 i = 0,j = 1,k = 2 且 l = 4时,有nums[i] < nums[k] < nums[j] < nums[l]。
    示例2
    输入:nums = [1,2,3,4]
    输出:0
    解释:只存在一个四元组 i = 0,j = 1,k = 2,l = 3但不满足nums[j]<nums[k],所以返回0。

提示
4 <= nums.length <= 4000
1 <= nums[i] <= nums.lenth
nums中所有数字互不相同,nums是一个排列。

2. 解释

首先写4个数,就1324这就是一个满足条件的四元组。nums[i]<nums[k]<nums[j]<nums[l]
例子

这是一个满足条件的四元组,透过这个例子我们可以看出来什么呢?
现在有两个选择:固定j和k或者固定i和l
首先先选择固定i和l,当我们固定了i和l就代表了,j和k就要在i和l中间寻找,寻找大于nums[i]且小于nums[l]的j和k,这其实也不是很难用前缀和找就可以,但是找到了又不能直接用,因为还有满足nums[k]<nums[j]这个条件,如果我们选择固定j和k就不一样了。

选择固定j和k,选取的i和k肯定都是满足条件的i和k,当这一条件满足时,直接找i和l就方便多了。我们只需要找到小于nums[k]且在j位置前的数据个数,和找到大于nums[j]且在k位置后的数据个数。

于是可以把great[k][x]表示为:在k右侧比 x = nums[j]大的元素个数。
把less[j][x]表示为:在j左侧比x = nums[k]小的元素个数。
那么解决方法就出来了,枚举j和k,把满足大小关系的i的个数和l的个数。

less[j][nums[k]]*great[k][nums[j]].

3. DP前缀和+枚举

考虑动态规划:

  • 如果x>=nums[k+1],那么[k右边的比x大的数]就会等于[k+1右边的比x大的数],也就是great[k][x] = great[k+1][x]。
    8 4 6 5 7 9 3 2 1为例,当前的4元组为4 6 5 7
    因为x大于nums[k+1] = 7,那么对于大于k右边的比x大的数,k+1的位置就是无关变量。
    例子

  • 如果x<nums[k+1],那么额外加1,也就是great[k][x] = great[k+1][x]+1。
    这个也很好理解,x比nums[k+1]小,nums[k+1]也占一个大于x的位置,那great[k][x]就会在加1咯。
    也就可以整理得到:

if(x>=nums[k+1])great[k][x] = great[k+1][x];
elsegreat[k][x] = great[k+1][x]+1;

因为在求great[k][x]前需要知道great[k+1][x]的信息,也就确定的我们需要倒序遍历数组。
因为还有数组数据是1~n的,在处理x的问题上,直接在循环里嵌套一个x从n到1的循环就可以了。当然这个肯定是可以优化的。
对于less的计算是同理的:

if(x<=nums[j-1])less[j][x] = less[j-1][x]
elseless[j][x] = less[j-1][x]+1
class Solution {
public:long long countQuadruplets(vector<int>& nums) {int n = nums.size();long long ans = 0;vector<vector<long long>> great(n,vector<long long>(n+1)),less(n,vector<long long>(n+1));for(int k = n-2;k>=0;--k){for(int x = n;x>=1;--x){if(x>=nums[k+1]){great[k][x] = great[k+1][x];}else{great[k][x] = great[k+1][x]+1;}}}for(int j = 1;j<n;++j){for(int x = 1;x<=n;++x){if(x<=nums[j-1]){less[j][x] = less[j-1][x];}else{less[j][x] = less[j-1][x]+1;}}}for(int j = 1;j<n;++j){for(int k = j+1;k<n;++k){if(nums[k]<nums[j]){ans+=(less[j][nums[k]]*great[k][nums[j]]);}}}return ans;}
};

后续可能会优化代码吧。


http://www.ppmy.cn/news/1523840.html

相关文章

Redis-基础篇

Redis简单介绍 Redis是一种键值型的NoSql数据库&#xff0c;这里有两个关键字&#xff1a; 键值型NoSql 其中键值型&#xff0c;是指Redis中存储的数据都是以key.value对的形式存储&#xff0c;而value的形式多种多样&#xff0c;可以是字符串.数值.甚至json&#xff1a; 而…

用于完善智能电表设计的 FPGA 到 ASIC 研究

许多嵌入式系统设计首先使用 FPGA 来实现。这可能是为了更快地进行原型设计或提供软件开发平台。有时&#xff0c;生产开始后&#xff0c;FPGA 仍保留在设计中。但通常情况下&#xff0c;计划是将 FPGA&#xff08;或多个 FPGA&#xff09;转换为 ASIC 以进行批量制造。 人们很…

iOS——持久化

iOS的数据存储机制 沙盒机制 应用沙盒文件夹包含了&#xff1a; Application(应用程序包)&#xff1a;包含了所有的资源文件和和可执行文件&#xff0c;上架前经过数字签名&#xff0c;上架后不可修改。 Documents&#xff1a;文档目录&#xff0c;要保存程序生成的数据&…

COD论文笔记 BiRefNet

本质还是一个 U 型编码器解码器结构的分割模型。 我可以考虑将©和(d)结合&#xff0c;即对解码器的输入不进行 patchify,同时在各个阶段引入梯度参考信息 最近的相关工作&#xff0c;中间监督、额外先验(频率&#xff0c;梯度&#xff0c;边缘等)取得不错效果 作者观察到…

Java健康养老智慧相伴养老护理小程序系统源码代办陪诊陪护更安心

健康养老&#xff0c;智慧相伴 —— 养老护理小程序&#xff0c;代办陪诊陪护更安心 &#x1f308;【开篇&#xff1a;智慧养老&#xff0c;新时代的温馨守护】&#x1f308; 在这个快节奏的时代&#xff0c;我们总希望能给予家人更多的关爱与陪伴&#xff0c;尤其是家中的长…

Go语言实战 pdf

这本书更加注重Go 语言的实战技能&#xff0c;Go语言结合了底层系统语言的能力以及现代语言的高级特性&#xff0c;旨在降低构建简单、可靠、高效软件的门槛。本书向读者提供一个专注、全面且符合语言习惯的视角&#xff0c;这是一本不错的Go 语言入门书。 百度网盘分享

pv和pvc自动匹配、自动创建、自动挂载

目录 概念 pv的状态 pvc在请求的过程中支持的权限控制选项 pv的回收策略 Retain 保留 Delete 删除 Recycle 回收 在yaml文件中指定pv的回收策略 静态pv 1. 配置NFS 2. 创建pv 3. 创建pvc 动态pv 动态pv的步骤 1.配置NFS 2.创建角色、赋权、绑定角色 3.创建NFS…

Java 中处理 XML 文件

在 Java 中处理 XML 文件&#xff0c;通常使用两种主要的解析方式&#xff1a;DOM 解析 和 SAX 解析。每种解析方式各有优劣&#xff0c;适用于不同的场景。下面详细解释这两种 XML 解析方法的基本原理、适用场景、共性规律、注意事项和特殊技巧。 1. DOM 解析 (Document Obje…