为了账号安全,请及时绑定邮箱和手机立即绑定

差分数组详解

一维差分数组

假设给你一个数组 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] ,如下图所示。

https://img1.sycdn.imooc.com//6482b63600019e1b10030338.jpg

我们可以看到原数组就是差分数组的前缀和。

 nums[0] = d[0]
 num[3] = d[0] + d[1] + d[2] + d[3]

有了差分数组,如果对区间 [a,b] 每个元素加 3 ,不需要在一个个操作,只需要在两端修改即可,如下图所示。

 d[a] += 3;
 d[b+1] -= 3;

https://img1.sycdn.imooc.com//6482b6370001526f06230282.jpg

来看下代码:

 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]

https://img1.sycdn.imooc.com//6482b6370001725502960250.jpg

如果想获取原数组,根据上面的公式可以得到:

 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
 }

https://img1.sycdn.imooc.com//6482b63700011b5c11610544.jpg

代码如下:

 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;
 }



点击查看更多内容
1人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消