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

一维前缀和与二维前缀和

前缀和(Prefix Sum)是一种用于高效计算数组区间和的预处理技术,尤其适用于需要频繁查询子数组或子矩阵和的场景。下面详细讲解一维前缀和与二维前缀和的原理、构建方法及应用。


一、一维前缀和

1. 定义
  • 前缀和数组 prefix 的每个元素 prefix[i] 表示原数组 arri 个元素的和(通常从 arr[0]arr[i-1])。
  • 例如,原数组 arr = [1, 2, 3, 4],前缀和数组为 prefix = [0, 1, 3, 6, 10]
2. 构建方法
  • 初始化 prefix[0] = 0
  • 递推公式:
    [
    \text{prefix}[i] = \text{prefix}[i-1] + \text{arr}[i-1]
    ]
  • 代码实现
    vector<int> buildPrefix(vector<int>& arr) {int n = arr.size();vector<int> prefix(n + 1, 0);for (int i = 1; i <= n; i++) {prefix[i] = prefix[i - 1] + arr[i - 1];}return prefix;
    }
    
3. 查询区间和
  • 查询区间 [L, R] 的和(左闭右闭区间):
    [
    \text{sum}(L, R) = \text{prefix}[R+1] - \text{prefix}[L]
    ]
  • 示例
    arr = [1, 2, 3, 4],求 [1, 2] 的和:
    [
    \text{sum}(1, 2) = \text{prefix}[3] - \text{prefix}[1] = 6 - 1 = 5
    ]
4. 应用场景
  • 快速计算子数组的和(时间复杂度 O(1))。
  • 解决滑动窗口问题(如和大于等于目标值的最短子数组)。

二、二维前缀和

1. 定义
  • 二维前缀和数组 prefix 的每个元素 prefix[i][j] 表示从矩阵左上角 (0,0) 到右下角 (i-1,j-1) 的子矩阵的和。
  • 例如,矩阵 matrix = [[1,2],[3,4]],前缀和数组为:
    [
    \text{prefix} = \begin{bmatrix}
    0 & 0 & 0 \
    0 & 1 & 3 \
    0 & 4 & 10 \
    \end{bmatrix}
    ]
2. 构建方法
  • 初始化 prefix[0][0] = 0
  • 递推公式:
    [
    \text{prefix}[i][j] = \text{prefix}[i-1][j] + \text{prefix}[i][j-1] - \text{prefix}[i-1][j-1] + \text{matrix}[i-1][j-1]
    ]
  • 代码实现
    vector<vector<int>> build2DPrefix(vector<vector<int>>& matrix) {int m = matrix.size();int n = matrix[0].size();vector<vector<int>> prefix(m + 1, vector<int>(n + 1, 0));for (int i = 1; i <= m; i++) {for (int j = 1; j <= n; j++) {prefix[i][j] = prefix[i-1][j] + prefix[i][j-1] - prefix[i-1][j-1] + matrix[i-1][j-1];}}return prefix;
    }
    
3. 查询子矩阵和
  • 查询子矩阵 (x1, y1)(x2, y2) 的和(左闭右闭区间):
    [
    \text{sum}(x1, y1, x2, y2) = \text{prefix}[x2+1][y2+1] - \text{prefix}[x1][y2+1] - \text{prefix}[x2+1][y1] + \text{prefix}[x1][y1]
    ]
  • 示例
    matrix = [[1,2,3],[4,5,6],[7,8,9]],求子矩阵 (1,1)(2,2) 的和:
    [
    \text{sum} = 5 + 6 + 8 + 9 = 28 \
    \text{通过前缀和计算:} \text{prefix}[3][3] - \text{prefix}[1][3] - \text{prefix}[3][1] + \text{prefix}[1][1] = 45 - 6 - 12 + 1 = 28
    ]
4. 应用场景
  • 快速计算子矩阵的和(时间复杂度 O(1))。
  • 图像处理中的区域像素和统计。
  • 动态规划中的矩阵路径问题。

三、对比总结

特性一维前缀和二维前缀和
数据结构一维数组二维数组
构建复杂度O(n)O(mn)
查询复杂度O(1)O(1)
核心公式prefix[i] = prefix[i-1] + arr[i-1]prefix[i][j] = ...(见上文)
应用问题子数组和、滑动窗口子矩阵和、图像处理、动态规划

四、经典例题

  1. 一维前缀和

    • LeetCode 303. 区域和检索 - 数组不可变
    • LeetCode 560. 和为 K 的子数组
  2. 二维前缀和

    • LeetCode 304. 二维区域和检索 - 矩阵不可变
    • LeetCode 1292. 元素和小于等于阈值的正方形的最大边长

通过掌握前缀和和二维前缀和的原理与实现,可以高效解决许多与区间和相关的算法问题。

http://www.lryc.cn/news/536143.html

相关文章:

  • 3×2 MIMO系统和2×2 MIMO系统对比
  • 【MySQL — 数据库基础】深入解析 MySQL 的联合查询
  • 【医院运营统计专题】3.解码医院运营统计:目标、原则与未来蓝图
  • Ubuntu 下 nginx-1.24.0 源码分析 - ngx_atomic_cmp_set 函数
  • CNN-BiLSTM卷积神经网络双向长短期记忆神经网络多变量多步预测,光伏功率预测
  • 【YOLO系列】YOLOv5 NMS源码理解、更换为DIoU-NMS
  • Android RenderEffect对Bitmap高斯模糊(毛玻璃),Kotlin(1)
  • 【linux学习指南】线程同步与互斥
  • JavaScript函数与方法详解
  • 【论文笔记】ZeroGS:扩展Spann3R+GS+pose估计
  • AtCoder - arc058_d Iroha Loves Strings解答与注意事项
  • 企业使用统一终端管理(UEM)工具提高端点安全性
  • Leetcode 算法题 9 回文数
  • 设计模式Python版 命令模式(上)
  • C语言之循环结构:直到型循环
  • 细说STM32F407单片机RTC的备份寄存器原理及使用方法
  • MATLAB计算反映热需求和能源消耗的度数日指标(HDD+CDD)(全代码)
  • J6 X8B/X3C切换HDR各帧图像
  • 09-轮转数组
  • 用vue3写一个好看的wiki前端页面
  • 瑞芯微烧写工具
  • 说下JVM中一次完整的GC流程?
  • Open FPV VTX开源之OSD使用分类
  • 智慧农业-虫害及生长预测
  • Python 识别图片和扫描PDF中的文字
  • C语言如何知道当前系统中的编译器数据类型的大小是多少?
  • gitlab Webhook 配置jenkins时“触发远程构建 (例如,使用脚本)”报错
  • Mysql中使用sql语句生成雪花算法Id
  • /etc/profile vs ~/.bashrc:如何正确使用?
  • SpringBoot实战:高效获取视频资源