差分数组详解
2023/6/9 14:22:09
本文主要是介绍差分数组详解,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
一维差分数组
假设给你一个数组 nums ,先对区间 [a,b] 中每个元素加 3 ,在对区间 [c,d] 每个元素减 5 …… ,这样非常频繁的区间修改,常规的做法可以一个个计算。
public void increment(int[] nums, int a, int b, int k) { for (int index = a; index <= b; index++) { nums[index] += k; } }
频繁对数组的一段区间进行增加或减去同一个值,如果一个个去操作,很明显效率很差,我们可以使用差分数组,差分数组就是原始数组相邻元素之间的差。定义差分数组 d[n] ,我们可以得到: d[i] = nums[i] − nums[i−1] ,其中 d[0] = nums[0] ,如下图所示。
我们可以看到原数组就是差分数组的前缀和。
nums[0] = d[0] num[3] = d[0] + d[1] + d[2] + d[3]
有了差分数组,如果对区间 [a,b] 每个元素加 3 ,不需要在一个个操作,只需要在两端修改即可,如下图所示。
d[a] += 3; d[b+1] -= 3;
来看下代码:
public class DiffNums { private int[] diff;// 差分数组。 private int[] nums;// 原数组。 public DiffNums(int[] nums) { this.nums = nums; diff = new int[nums.length]; diff[0] = nums[0]; for (int i = 1; i < diff.length; i++) diff[i] = nums[i] - nums[i - 1]; } // 给区间[a,b]每个元素增加val(也可为负数)。 public void increment(int a, int b, int val) { diff[a] += val; if (b + 1 < diff.length) diff[b + 1] -= val; } // 返回结果数组。 public int[] result() { nums[0] = diff[0]; for (int i = 1; i < diff.length; i++) nums[i] = diff[i] + nums[i - 1]; return nums; } }
二维差分数组
我们把一维差分数组看做是一条直线,只需要用后面的值减去前面的值就可以构造差分数组。而二维差分数可以把他看做是一个平面,如下图所示,他的定义如下:
d[i][j] = a[i][j]-a[i-1][j]-a[i][j-1]+a[i-1][j-1]
如果想获取原数组,根据上面的公式可以得到:
a[i][j] = a[i-1][j]+a[i][j-1]-a[i-1][j-1]+d[i][j]
如下图所示,以 (x1,y1) 为左上角, (x2,y2) 为右下角构成一个区间,如果对这个区间内的每个元素增加 val ,只需要执行下面四步即可。
public void add(int x1, int y1, int x2, int y2, int val) { d[x1][y1] += val;// 增加S1 d[x1][y2 + 1] -= val;// 减去S2 d[x2 + 1][y1] -= val;// 减去S3 d[x2 + 1][y2 + 1] += val;//加上S4 }
代码如下:
private int[][] d;// 差分数组。 private int[][] a;// 原数组。 public TwoDiffNums(int[][] a) { this.a = a; int m = a.length; int n = a[0].length; d = new int[m][n]; // 求差分数组。 for (int i = 0; i < m; i++) for (int j = 0; j < n; j++) add(i, j, i, j, a[i][j]); } public void add(int x1, int y1, int x2, int y2, int val) { d[x1][y1] += val; if (y2 + 1 < d[0].length) d[x1][y2 + 1] -= val; if (x2 + 1 < d.length) d[x2 + 1][y1] -= val; if (x2 + 1 < d.length && y2 + 1 < d[0].length) d[x2 + 1][y2 + 1] += val; } // 返回结果数组。 public int[][] result() { for (int i = 0; i < a.length; i++) { for (int j = 0; j < a[0].length; j++) { int x1 = i > 0 ? a[i - 1][j] : 0; int x2 = j > 0 ? a[i][j - 1] : 0; int x3 = i > 0 && j > 0 ? a[i - 1][j - 1] : 0; a[i][j] = x1 + x2 - x3 + d[i][j]; } } return a; }
这篇关于差分数组详解的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2025-01-102025 蛇年,J 人直播带货内容审核团队必备的办公软件有哪 6 款?
- 2025-01-10高效运营背后的支柱:文档管理优化指南
- 2025-01-10年末压力山大?试试优化你的文档管理
- 2025-01-10跨部门协作中的进度追踪重要性解析
- 2025-01-10总结 JavaScript 中的变体函数调用方式
- 2025-01-10HR团队如何通过数据驱动提升管理效率?6个策略
- 2025-01-10WBS实战指南:如何一步步构建高效项目管理框架?
- 2025-01-10实现精准执行:团队协作新方法
- 2025-01-10如何使用工具提升活动策划团队的工作效率?几个必备工具推荐
- 2025-01-10WiX 标签使用介绍:打造专业安装程序的利器