给定一个二维矩阵 matrix,以下类型的多个请求:

计算其子矩形范围内元素的总和,该子矩阵的 左上角 为 (row1, col1) ,右下角 为 (row2, col2) 。
实现 NumMatrix 类:

NumMatrix(int[][] matrix) 给定整数矩阵 matrix 进行初始化
int sumRegion(int row1, int col1, int row2, int col2) 返回 左上角 (row1, col1) 、右下角 (row2, col2) 所描述的子矩阵的元素 总和 。

示例 1: image.png

输入:
[“NumMatrix”,”sumRegion”,”sumRegion”,”sumRegion”]
[[[[3,0,1,4,2],[5,6,3,2,1],[1,2,0,1,5],[4,1,0,1,7],[1,0,3,0,5]]],[2,1,4,3],[1,1,2,2],[1,2,2,4]]
输出:
[null, 8, 11, 12]

解释:
NumMatrix numMatrix = new NumMatrix([[3,0,1,4,2],[5,6,3,2,1],[1,2,0,1,5],[4,1,0,1,7],[1,0,3,0,5]]);
numMatrix.sumRegion(2, 1, 4, 3); // return 8 (红色矩形框的元素总和)
numMatrix.sumRegion(1, 1, 2, 2); // return 11 (绿色矩形框的元素总和)
numMatrix.sumRegion(1, 2, 2, 4); // return 12 (蓝色矩形框的元素总和)

提示:

m == matrix.length
n == matrix[i].length
1 <= m, n <= 200
-10^5 <= matrix[i][j] <= 10^5
0 <= row1 <= row2 < m
0 <= col1 <= col2 < n
最多调用 10^4 次 sumRegion 方法

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/range-sum-query-2d-immutable
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路

使用二维数组preSum,304. 二维区域和检索 - 矩阵不可变 - 图2表示「以[0,0]为左上角,[i,j]为右下角的矩阵」的和。

具体实现时和一维前缀和数组一样,两个维度长度均加一。

计算304. 二维区域和检索 - 矩阵不可变 - 图3矩阵对应的304. 二维区域和检索 - 矩阵不可变 - 图4时,可以转化为304. 二维区域和检索 - 矩阵不可变 - 图5。其中A坐标为[0,0],E为[i,j],C为E相邻的上面的点,D为左边的,B为左上方的。

写成代码为304. 二维区域和检索 - 矩阵不可变 - 图6

image.png

计算区域和时也可以转换,例如下图304. 二维区域和检索 - 矩阵不可变 - 图8304. 二维区域和检索 - 矩阵不可变 - 图9表示矩阵304. 二维区域和检索 - 矩阵不可变 - 图10的区域和,而等号右边的四个矩阵都可以通过304. 二维区域和检索 - 矩阵不可变 - 图11304. 二维区域和检索 - 矩阵不可变 - 图12#card=math&code=O%281%29&id=uNbmq)时间获得

image.png

代码

  1. class NumMatrix {
  2. int[][] preSum;
  3. public NumMatrix(int[][] matrix) {
  4. int n = matrix.length;
  5. int m = matrix[0].length;
  6. preSum = new int[n + 1][m + 1];
  7. for (int i = 1; i <= n; i++) {
  8. for (int j = 1; j <= m; j++) {
  9. preSum[i][j] = preSum[i][j - 1] + preSum[i - 1][j] - preSum[i - 1][j - 1] + matrix[i - 1][j - 1];
  10. }
  11. }
  12. }
  13. public int sumRegion(int row1, int col1, int row2, int col2) {
  14. // 对应上面的AF-AC-AD+AB
  15. return preSum[row2 + 1][col2 + 1] - preSum[row1][col2 + 1] - preSum[row2 + 1][col1] + preSum[row1][col1];
  16. }
  17. }
  18. /**
  19. * Your NumMatrix object will be instantiated and called as such:
  20. * NumMatrix obj = new NumMatrix(matrix);
  21. * int param_1 = obj.sumRegion(row1,col1,row2,col2);
  22. */