使用前缀和数组解决“区间和查询“问题

news2024/11/26 20:48:26

本文已收录到 GitHub · AndroidFamily,有 Android 进阶知识体系,欢迎 Star。技术和职场问题,请关注公众号 [彭旭锐] 进 Android 面试交流群。

前言

大家好,我是小彭。

今天分享到一种非常有趣的数据结构 —— 前缀和数组。前缀和的思想本身很容易理解,同时也是理解更高难度的线段树、字典树等数据结构的基础。

那么,什么是前缀和,我们可以使用前缀和解决什么问题呢?今天我们就围绕这两个问题展开。


学习路线图:


1. 什么是前缀和

前缀和数组是一种用来高效地解决 “静态数据的频繁区间和查询” 问题的数据结构。

先举个例子,给定一个整数数组,要求输出数组在 [ i , j ] [i, j] [i,j] 区间内元素的总和,这就是区间查询问题。这道题的暴力解法是很容易想到的,无非就是把 [ i , j ] [i, j] [i,j] 区间中所有元素累计而已即可,时间复杂度是 O ( n ) O(n) O(n),空间复杂度是 O ( 1 ) O(1) O(1)

单次查询确实已经没有优化空间了,但如果进行频繁的区间和查询,很明显会有非常多重复的求和运算。例如,在查询 [ 1 , 5 ] [1, 5] [1,5] 区间和 [ 1 , 10 ] [1, 10] [1,10] 区间时, [ 1 , 5 ] [1, 5] [1,5] 这个子区间就被重复计算了。那么,有可能借助 “空间换时间”,在 O ( 1 ) O(1) O(1) 时间复杂度内计算 [ 5000 , 1000000 ] [5000,1000000] [5000,1000000] 上的区间和吗?

这就需要使用前缀和 + 差分技巧:

  • 预处理阶段: 开辟一个前缀和数组,记录每个位置到所有前驱节点的累加和 p r e S u m preSum preSum,总共需要 O(n) 时间;
  • 查询阶段: 将区间左右端点的区间和相减,就可以在 O ( 1 ) O(1) O(1) 时间得到这个区间中所有元素的累加和,避免了每次查询都需要 for 循环遍历整个区间求和。例如,要求 [ i , j ] [i, j] [i,j] 区间的和,就是直接用 p r e S u m [ j + 1 ] − p r e S u m [ i ] preSum[j + 1] - preSum[i] preSum[j+1]preSum[i] 获得。

前缀和示意图


2. 典型例题 · 区间和检索

理解以上概念后,就已经具备解决区间和问题的基本知识了。我们来看一道 LeetCode 上的前缀和典型例题:LeetCode 303. 区域和检索 - 数组不可变

LeetCode 例题

题解

class NumArray(nums: IntArray) {

    // 前缀和数组
    // 数组长度加一后不用考虑数组越界,代码更简洁
    private val preSum = IntArray(nums.size + 1) { 0 }

    init {
        for (index in nums.indices) {
            preSum[index + 1] = preSum[index] + nums[index]
        }
    }

    fun sumRange(i: Int, j: Int): Int {
        return preSum[j + 1] - preSum[i]
    }
}

代码很简单,其中前缀和数组 preSum 的长度要额外加 1 是为了简化数组越界判断。我们来分析它的复杂度:

  • 时间复杂度: 构建前缀和数组的时间复杂度是 O ( n ) O(n) O(n),查询的时间复杂度是 O ( m ) O(m) O(m) n n n 是数据量, m m m 是区间和查询的次数;
  • 空间复杂度: O ( n ) O(n) O(n),使用了 长度为 n + 1 n+ 1 n+1 的前缀和数组。

另外,前缀和还适用于二维区间和检索,思路都是类似的,你可以试试看: LeetCode · 304. 二维区域和检索 - 矩阵不可变


3. 典型例题 · 前缀和 + 哈希表

继续看另一道前缀和与哈希表结合的例题:LeetCode 560. 和为 K 的子数组

LeetCode 例题

这道题就是在考前缀和思想,我们可以轻松地写出第一种解法:

题解

class Solution {
    fun subarraySum(nums: IntArray, k: Int): Int {
        // 1、预处理:构造前缀和数组
        var preSum = IntArray(nums.size + 1) { 0 }
        for (index in nums.indices) {
            preSum[index + 1] = preSum[index] + nums[index]
        }

        // 2、枚举所有子数组,使用「前缀和 + 差分」技巧计算区间和
        var result = 0
        for (i in nums.indices) {
            for (j in i until nums.size) {
                val sum_i_j = preSum[j + 1] - preSum[i]
                if (k == sum_i_j) {
                    result++
                }
            }
        }
        return result
    }
}

在这个题解中,我们枚举每个子数组,使用「前缀和 + 差分」技巧计算区间和为 K 的个数,我们来分析下它的复杂度:

  • 时间复杂度: 一共存在 n 2 n^2 n2 个子数组,单次子数组的区间查询是 O ( 1 ) O(1) O(1),所以整体的时间复杂度是 O ( n 2 ) O(n^2) O(n2)
  • 空间复杂度: O ( n ) O(n) O(n)

时间复杂度有办法优化吗?我们发现,题目要求的是数组个数,而不关心具体的数组,所以我们不必枚举全部子数组(一共有 n 2 n^2 n2 个子数组), 我们只需要在计算出当前位置的前缀和之后,观察之前的位置中是否存在前缀和正好等于 p r e S u m [ j ] − K preSum[j] - K preSum[j]K 的位置,有的话,就说明它们之间的区间和就是 K K K 把满足条件的个数累加起来,就是最终结果。

前缀和示意图

紧接着另一个问题是:怎么快速找到前缀和等于 p r e S u m [ j ] − K preSum[j] - K preSum[j]K 的位置?聪明的你一定知道了—— 哈希表。 我们可以维护一个哈希表,记录前缀和出现的位置,就可以用 O(1) 找到它。 由于前缀和有可能会重复出现,而且我们只关心次数不关心位置,所以映射关系应该为 <前缀和 - 出现次数>。

题解

class Solution {
    fun subarraySum(nums: IntArray, k: Int): Int {
        var preSum = 0
        var result = 0

        // 维护哈希表<前缀和,出现次数>
        val map = HashMap<Int, Int>()
        map[0] = 1

        for (index in nums.indices) {
            preSum += nums[index]
            // 获得前缀和为 preSum - k 的出现次数
            val offset = preSum - k
            if (map.contains(offset)) {
                result += map[offset]!!
            }
            map[preSum] = map.getOrDefault(preSum, 0) + 1
        }
        return result
    }
}

我们来分析下它的复杂度:

  • 时间复杂度: 每个元素只处理一次,整体时间复杂度是 O ( n ) O(n) O(n)
  • 空间复杂度: O ( n ) O(n) O(n)

4. 典型例题 · 前缀和 + 单调队列

继续看一道前缀和与单调队列结合的例题,你可以先做一下第 53 题:

  • LeetCode 53. 最大子数组和
  • LeetCode 918. 环形子数组的最大和

LeetCode 例题

在第 53 题中,我们只需要维护一个当前观察到的最小前缀和变量,将其与当前的前缀和做差值,就可以得到以当前节点为右端点的最大的区间和。这一题就是考前缀和思想,相对简单。

第 53 题题解

class Solution {
    fun maxSubArray(nums: IntArray): Int {
        // 前缀和 + 单调:维护最小的前缀和
        var minPreSum = 0
        var result = Integer.MIN_VALUE
        var preSum = 0

        for (element in nums) {
            preSum += element
            result = Math.max(result, preSum - minPreSum)
            minPreSum = Math.min(minPreSum, preSum)
        }
        return result
    }
}

在第 918 题中,数组变为环形数组,环形数组的问题一般都会用 2 倍的 “假数据长度” 做模拟,求前缀和数组这一步大同小异。

关键在于: “子数组最多只能包含固定缓冲区 num 中的每个元素一次”,这意味随着观察的区间右节点逐渐向右移动,所允许的左区间会限制在一个滑动窗口范围内,以避免元素重复出现。因此,一个变量不再能够满足题目需求。

示意图

所以我们的问题就是要求这个 “滑动窗口中的最小前缀和”。Wait a minute! 滑动窗口的最小值?这不就是 使用单调队列解决 “滑动窗口最大值” 问题 这篇文章讲的内容吗,秒懂,单调队列安排一下。

第 918 题题解

class Solution {
    fun maxSubarraySumCircular(nums: IntArray): Int {
        val preSum = IntArray(nums.size * 2 + 1).apply {
            for (index in 0 until nums.size * 2) {
                this[index + 1] = this[index] + nums[index % nums.size]
            }
        }

        // 单调队列(从队头到队尾递增)
        val queue = LinkedList<Int>()
        var result = Integer.MIN_VALUE

        for (index in 1 until preSum.size) {
            // if:移除队头超出滑动窗口范围的元素
            // 前缀和窗口 k 为 length + 1,比原数组上的逻辑窗口大 1 位,因为区间的差值要找前一个节点的前缀和
            if (!queue.isEmpty() && queue.peekFirst() < index - nums.size /* index - k + 1 */) {
                queue.pollFirst()
            }

            // 从队头取目标元素
            result = Math.max(result, preSum[index] - (preSum[queue.peekFirst() ?: 0]))

            // while:队尾元素大于当前元素,说明队尾元素不再可能是最小值,后续不再考虑它
            while (!queue.isEmpty() && preSum[queue.peekLast()] >= preSum[index]) {
                // 抛弃队尾元素
                queue.pollLast()
            }
            queue.offerLast(index)
        }
        return result
    }
}

我们来分析它的时间复杂度:

  • 时间复杂度: 构建前缀和数组 O ( n ) O(n) O(n),前缀和数组中每个元素在单调队列中入队和出队 1 1 1 次,因此整体时间复杂度是 O ( n ) O(n) O(n);
  • 空间复杂度: 构建前缀和数组 O ( n ) O(n) O(n),最坏情况下(递减序列)所有元素都被添加到单调队列中,因此整体空间复杂度是 O ( n ) O(n) O(n)

提示: 这道题还有使用 Kadane 算法 O ( 1 ) O(1) O(1) 空间复杂度解法。


5. 总结

到这里,前缀和的内容就讲完了。文章开头也提到了, 前缀和数组是一种高效地解决静态数据的频繁区间和查询问题的数据结构,只需要构造一次前缀和数组,就能使用 O(1) 时间完成查询操作。

那么,在动态数据的场景中(即动态增加或删除元素),使用前缀和数组进行区间和查询是否还保持高效呢?如果不够高效,有其他的数据结构可以解决吗?你可以思考以下 2 道题:

  • LeetCode · 307. 区域和检索 - 数组可修改
  • LeetCode · 308. 二维区域和检索 - 矩阵不可变

更多同类型题目:

前缀和难度题解
303. 区间和检索 - 数组不可变Easy【题解】
724. 寻找数组的中心下标Easy【题解】
304. 二维区域和检索 - 矩阵不可变Medium【题解】
560. 和为 K 的子数组Medium【题解】
974. 和可被 K 整除的子数组Medium【题解】
1314. 矩阵区域和Medium【题解】
918. 环形子数组的最大和Medium【题解】
525. 连续数组Medium【题解】
1248. 统计「优美子数组」Medium【题解】

参考资料

  • LeetCode 专题 · 前缀和 —— LeetCode 出品
  • LeetCode 题解 · 560. 和为 K 的子数组 —— liweiwei1419 著
  • 小而美的算法技巧:前缀和数组 —— labuladong 著
  • 小而美的算法技巧:差分数组 —— labuladong 著

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

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

相关文章

每日一题|2022-11-8|1684. 统计一致字符串的数目|哈希表|Golang

1684. 统计一致字符串的数目 思路1:丢人做法 哈希记录allowed&#xff0c;暴力遍历words所有字母&#xff0c;如果有不在哈希表里的&#xff0c;计数。最后用words的长度减去 计数 就行。 func countConsistentStrings(allowed string, words []string) int {has1 : make(map[…

如何判断一段程序是否是裸机程序?

在嵌入式MCU领域&#xff0c;一般将不移植操作系统直接烧录运行的程序称为裸机程序。 一般来说&#xff0c;非易失性存储&#xff0c;时钟&#xff0c;图形显示&#xff0c;网络通讯&#xff0c;用户I/O设备…都需要硬件依赖。 基于硬件基础&#xff0c;内存管理、文件系统、…

【API部署】fastapi与nuitka打包py项目

提示&#xff1a;分两部分&#xff1a;fastapi接口调用&#xff0c;与nuitka快速打包 功能&#xff1a;作为一名算法工程师&#xff0c;训练机器学习模型只是为客户提供解决方案的一部分。 除了生成和清理数据、选择和调整算法之外&#xff0c;还需交付和部署结果&#xff0c;…

130道基础OJ编程题之: 29 ~ 38 道

130道基础OJ编程题之: 29 ~ 38 道 文章目录130道基础OJ编程题之: 29 ~ 38 道0. 昔日OJ编程题:29. BC23 时间转换30. BC24 总成绩和平均分计算31. BC30 KiKi和酸奶32. BC31 发布信息33. BC3 输出学生信息34. BC33 计算平均成绩35. BC34 进制AB36. BC37 网购37.BC39 争夺前五名38…

【谷粒商城】

一、项目介绍 1.微服务架构图 2.微服务划分图 二、环境搭建 1.虚拟机搭建环境 这里我买了华为云&#xff0c;没用虚拟机 华为云配置 2.Linux 安装docker docker文档&#xff1a;https://docs.docker.com/engine/install/centos/ # 1. 卸载之前的dockersudo yum remove d…

[MySql]初识数据库与常见基本操作

专栏简介 :MySql数据库从入门到进阶. 题目来源:leetcode,牛客,剑指offer. 创作目标:记录学习MySql学习历程 希望在提升自己的同时,帮助他人,,与大家一起共同进步,互相成长. 学历代表过去,能力代表现在,学习能力代表未来! 文章目录 前言 1.初识数据库 1.1 数据库概述 1.2 数据库…

mysql隔离级别RR下的行锁、临键锁、间隙锁详解及运用

一&#xff1a;mysql 锁的基本概念 锁&#xff1a;悲观锁、乐观锁 悲观锁&#xff1a;写锁 for update、读锁for share 写锁&#xff1a;只允许当前事务读写&#xff0c;其它事务全部等待&#xff0c;包括读取数据&#xff0c;锁的数据范围需要具体分析 读锁&#xff1a;允…

【前端】Vue+Element UI案例:通用后台管理系统-Echarts图表:折线图、柱状图、饼状图

文章目录目标代码数据改写为动态Echarts引入与html结构折线图&#xff1a;orderData柱状图&#xff1a;userData饼状图&#xff1a;videoData总效果总代码:Home.vue上一篇&#xff1a;【前端】VueElement UI案例&#xff1a;通用后台管理系统-Echarts图表准备&#xff1a;axios…

公司缺人自己搞了vue又搞koa,熬夜把架子搭起来

如果有一天&#xff0c;人手紧缺&#xff0c;自己搞了前端还要搞服务端&#xff0c;今天我们把这个项目架子搭起来&#xff0c;让前端同学也可以轻松全栈开火。 技多不压身&#xff0c;活儿多了可压身啊 目录 一、上午写VUE 1、 新建一个我们的伟大项目文件夹 2、用vscode打…

程序中断方式

中断的基本概念 程序中断是指在计算机执行现行程序的过程中&#xff0c;出现某些急需处理的异常情况或特殊请求&#xff0c;CPU暂时中止现行程序&#xff0c;而转去对这些异常情况或特殊请求进行处理&#xff0c;在处理完毕后CPU又自动返回到现行程序的断点处&#xff0c;继续…

c语言之“数组”初级篇

前言 牛牛又和大家见面了&#xff0c;本篇牛牛要讲的内容是c语言中有关数组的内容。 欢迎大家一起学习&#xff0c;共同进步。 目录前言数组一、一维数组1.1 一维数组的创建1.2 一维数组的初始化1.3 一维数组的应用1.4 一维数组的存储二、二维数组2.1 二维数组创建2.2 二维数…

MySQL的select语句

SQL概述 SQL背景知识 1946 年&#xff0c;世界上第一台电脑诞生&#xff0c;如今&#xff0c;借由这台电脑发展起来的互联网已经自成江湖。在这几十年里&#xff0c;无数的技术、产业在这片江湖里沉浮&#xff0c;有的方兴未艾&#xff0c;有的已经几幕兴衰。但在这片浩荡的波…

基于android的车辆违章停放执法移动APP(ssm+uinapp+Mysql)-计算机毕业设计

车辆违章停放执法移动APP的功能已基本实现&#xff0c;主要实现首页&#xff0c;个人中心&#xff0c;市民管理&#xff0c;警察管理&#xff0c;罚单信息管理&#xff0c;缴费通知管理&#xff0c;系统管理等功能的操作系统。 论文主要从系统的分析与设计、数据库设计和系统的…

【机器学习】回归的原理学习与葡萄酒数据集的最小二乘法线性回归实例

文章目录一&#xff0c;回归1.1回归分析的基本概念1.2线性回归1.3最小二乘法1.4一元(简单)线性回归模型1.4.1随机误差项(线性回归模型)的假定条件1.4.2参数的普通最小二乘估计(0LS)1.5葡萄酒数据集的最小二乘法线性回归实例一&#xff0c;回归 1.1回归分析的基本概念 回归分析…

前端一面经典vue面试题总结

一般在哪个生命周期请求异步数据 我们可以在钩子函数 created、beforeMount、mounted 中进行调用&#xff0c;因为在这三个钩子函数中&#xff0c;data 已经创建&#xff0c;可以将服务端端返回的数据进行赋值。 ​ 推荐在 created 钩子函数中调用异步请求&#xff0c;因为在…

受激拉曼散射计量【Stimulated-Raman-Scattering Metrology】(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️❤️&#x1f4a5;&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑…

单元测试的时候读不到resources.test中配置

背景 接手了几个老工程&#xff0c;跑单元测试的时候&#xff0c;发现数据库的配置总是走了dev环境&#xff0c; 原因是工程中分环境进行了db的配置 历史经验 指定本地环境 ActiveProfiles(“test”) 没有生效 解决 在pom文件中 新加如下配置 <build><!--单元测…

Java—类加载机制

类加载机制 我们多次提到了类加载器ClassLoader&#xff0c;本章就来详细讨论Java中的类加载机制与ClassLoader。 类加载器ClassLoader就是加载其他类的类&#xff0c;它负责将字节码文件加载到内存&#xff0c;创建Class对象。与之前介绍的反射、注解和动态代理一样&#xf…

奶制品数据可视化,去年全国奶制品产量高达3778万吨,同比增长7.1%

奶制品是生活中很常见的一种补充人体所需维生素和矿物质元素的重要食品&#xff0c;在生活中奶制品也是很常见的&#xff0c;食用最多的是牛奶。牛奶中含有非常丰富的钙质&#xff0c;睡前适当给孩子食用&#xff0c;可以补充孩子所需的钙质从而达到长高的效果。 很多小伙伴经常…

C++ 类和对象以及内存管理 练习错题总结

作者&#xff1a;小萌新 专栏&#xff1a;C初阶作业 简介&#xff1a;大二学生 希望能和大家一起进步 本篇博客介绍&#xff1a;对于我们上一周学的知识做一个总结 查缺补漏 C 类和对象以及内存管理练习类和对象 (上)拷贝构造函数类和对象&#xff08;中&#xff09;重载函数运…