[优选算法专题四.前缀和——NO.26二维前缀和]

题目链接:

二维前缀和

题目描述:

题目解析:


代码逐部分解析

读入原始数据
cpp 复制代码
int n = 0, m = 0, q = 0;
cin >> n >> m >> q;
vector<vector<int>> arr(n + 1, vector<int>(m + 1));
for (int i = 1; i <= n; i++) {
    for (int j = 1; j <= m; j++) {
        cin >> arr[i][j];
    }
}
  • n:二维数组的行数,m:列数,q:查询次数。
  • 定义二维数组 arr,大小为 (n+1)×(m+1),下标从 1 开始(方便前缀和计算,避免处理边界时的额外判断)。
  • 双层循环读入 n×m 个元素,存储到 arr 中。
预处理前缀和矩阵 dp
cpp 复制代码
vector<vector<long long>> dp(n + 1, vector<long long>(m + 1)); // 防止溢出
for (int i = 1; i <= n; i++) {
    for (int j = 1; j <= m; j++) {
        dp[i][j] = dp[i-1][j] + dp[i][j-1] + arr[i][j] - dp[i-1][j-1];
    }
}
  • dp 是前缀和矩阵,类型为 long long(避免多个整数累加时溢出)。
  • dp[i][j] 表示:以 arr[1][1] 为左上角、arr[i][j] 为右下角的矩形区域的所有元素之和。
  • 计算公式推导dp[i][j] = 上方区域和(dp[i-1][j]) + 左方区域和(dp[i][j-1]) + 当前元素(arr[i][j]) - 重复计算的左上角区域和(dp[i-1][j-1])。(画图理解更直观:上方和左方区域重叠的部分被加了两次,需要减去一次)。
处理查询并输出结果
cpp 复制代码
int x1 = 0, y1 = 0, x2 = 0, y2 = 0;
while (q--) {
    cin >> x1 >> y1 >> x2 >> y2;
    cout << dp[x2][y2] - dp[x1-1][y2] - dp[x2][y1-1] + dp[x1-1][y1-1] << endl;
}
  • 每次查询输入矩形的左上角坐标 (x1, y1) 和右下角坐标 (x2, y2)
  • 区域和计算公式 :目标区域和 = 大矩形和(dp[x2][y2]) - 上方多余区域(dp[x1-1][y2]) - 左方多余区域(dp[x2][y1-1]) + 重复减去的左上角区域(dp[x1-1][y1-1])。(同样通过画图可清晰理解:减去上方和左方后,左上角重叠部分被多减了一次,需要加回)。

总结

  • 时间复杂度 :预处理前缀和为 O(n*m),每次查询为 O(1),总复杂度为 O(n*m + q),适合大量查询的场景。
  • 空间复杂度O(n*m)(存储原始数组和前缀和矩阵)。
  • 关键点 :数组下标从 1 开始简化边界处理,使用 long long 避免整数溢出。
相关推荐
吃好睡好便好4 小时前
在Matlab中绘制横直方图
开发语言·学习·算法·matlab
仰泳之鹅4 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
x_yeyue7 小时前
三角形数
笔记·算法·数论·组合数学
念何架构之路8 小时前
Go语言加密算法
数据结构·算法·哈希算法
AI科技星8 小时前
《数学公理体系·第三部·数术几何》(2026 年版)
c语言·开发语言·线性代数·算法·矩阵·量子计算·agi
失去的青春---夕阳下的奔跑8 小时前
560. 和为 K 的子数组
数据结构·算法·leetcode
黎阳之光8 小时前
黎阳之光:以视频孪生重构智慧医院信息化,打造高标项目核心竞争力
大数据·人工智能·物联网·算法·数字孪生
丷丩9 小时前
三级缓存下MVT地图瓦片服务性能优化策略
算法·缓存·性能优化·gis·geoai-up
m0_629494739 小时前
LeetCode 热题 100-----25.回文链表
数据结构·算法·leetcode·链表
ʚ希希ɞ ྀ10 小时前
单词拆分----dp
算法