差分数组详解

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

alt

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

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

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

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

alt

来看下代码:

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]

alt

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

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
}

alt

代码如下:

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;
}
#面试题打卡学习##面试题##面试题库##差分数组#
常见数据结构介绍 文章被收录于专栏

主要介绍一些常见的数据结构,包括数组,滚动数组,差分数组,树状数组,单向链表,双向链表,循环链表,跳表,异或链表,队列,循环队列,双端队列,栈,散列表,堆,二叉树,二叉搜索树,AVL树,红黑树,字典树,哈夫曼树,线段树,笛卡尔树,图的介绍,图的遍历,Dijkstra算法,Bellman-Ford算法,SPFA算法,Floyd算法,Prim算法,Kruskal算法,Boruvka算法,拓扑排序等。

全部评论
差分数组的定义是什么?
点赞 回复 分享
发布于 2023-06-09 11:42 吉林
差分数组的优化方式有哪些?
点赞 回复 分享
发布于 2023-06-09 11:03 上海

相关推荐

04-02 11:59
河海大学 Java
【吐槽+面经】ThunderSoft&nbsp;Java岗多对多群面被碾压,整理Java高频真题!今天真的被ThunderSoft线上多对多群面狠狠羞辱了,全程心态爆炸,整理下面试真题给后面的兄弟避坑😭一、面试地狱开局-&nbsp;形式:2位面试官+2位求职者同场,轮流答题,对比感直接拉满-&nbsp;对手配置:4个月实习+省级大创项目,Java体系烂熟于心,不保研不考研,手握多份offer不去,跟我卷8k-9k的岗-&nbsp;我的状态:简历项目、技术回答全被衬托,近1/3问题答得模糊/卡壳,大半时间在听大佬滔滔不绝,硬生生滞留会议室1小时,后期明显感觉没戏,好几次想直接退会二、本次Java岗核心考察方向官方明确:通用业务+项目经历、Java技术体系掌握,全程深挖基础+项目落地三、结合记忆整理|Java面试官高频提问(印象真题)a.&nbsp;Java核心基础(必考)1.&nbsp;面向对象三大特性:继承、多态理解2.&nbsp;抽象类与接口区别、使用场景3.&nbsp;异常处理:常见异常(空指针、IO、数组越界)、&nbsp;try-catch-finally&nbsp;用法4.&nbsp;集合:Set特点与去重场景,底层实现逻辑(好像有红黑树什么的忘了)5.&nbsp;JDK基础概念及实现组件方式、IOC核心理解6.&nbsp;重写与重载的区别b.&nbsp;并发编程(这个是一个场景题,要你设计一个仓库可以收发货物,要你说出如何解决大量货物出库迸发问题,很多忘了)1.&nbsp;高并发场景如何保证数据一致性(给了个仓库场景)2.&nbsp;消息队列在高并发中的作用(削峰、异步)3.&nbsp;锁的作用、使用场景4.&nbsp;死锁产生条件、解决方法5.&nbsp;事务在高并发购票中的应用c.&nbsp;数据库1.&nbsp;多表设计(用户/乐队/演唱会/账户表)2.&nbsp;表间关联关系如何保证d.&nbsp;项目落地(必问)1.&nbsp;团队任务分配、协作模式2.&nbsp;项目难点&amp;amp;解决方案e.&nbsp;通用问题1.&nbsp;AI在刷题、论文阅读/复现中的使用2.&nbsp;个人优势3.&nbsp;保研/考研&amp;amp;职业规划四、血泪教训1.&nbsp;多对多群面心态最关键,别被对手带节奏,把自己会的讲清楚2.&nbsp;Java基础、项目细节必须背熟抠透,别像我一样卡壳3.&nbsp;提前准备高并发、事务、锁等场景题,面试官最爱问祝大家面试顺利,别再像我一样被碾压了🙏对面也问了很多问题,很多都是我没回答出来,然后面试官:“刚才问到他的某个问题,你答一下”,然后他答出来后就问另外方向的问题了,一直问到不会的就深挖。根据模糊记忆让豆包整理的,凑合着看吧。
查看18道真题和解析
点赞 评论 收藏
分享
评论
3
2
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务