差分数组模板

差分数组模板参考于labuladong:论那些小而美的算法技巧:差分数组一、什么时候使用差分数组呢?相信很多人都遇到过这类题:给定一个原数组长度为n,查询次数m,每次查询给定一个区间[l,r]和一个整数k,使得原数组介于[l,r]之间的元素同时增(或减)k输出最终的数组num[8,2,6,3,1]m=2131023注:第一次查询num=83741第二次查询num=1161041最终num=1

大家好,又见面了,我是你们的朋友全栈君。

参考于 labuladong: 论那些小而美的算法技巧:差分数组

一、什么时候使用差分数组呢?

相信很多人都遇到过这类题:
给定一个原数组长度为 n,查询次数 m ,
每次查询给定一个区间 [l ,r] 和一个整数 k , 使得原数组介于 [l ,r] 之间的元素同时 增 (或减) k
输出最终的数组

num[ 8 , 2 , 6 , 3 , 1 ] m = 2
1 3 1
0 2 3
注:
第一次查询 num = 8 3 7 4 1
第二次查询 num = 11 6 10 4 1
最终 num = 11 6 10 4 1

当然了,每次查询,遍历一下区间 [l ,r] 对其进行修改,结果肯定是对的
但是呢,笔试 和 刷题 时,如果数据给的比较大,比较严苛,多数是会超时,时间复杂度是 O(mn)

二、什么是差分数组 ?

这时就需要用到了差分数组的技巧来解答,
差分数组 : 主要适用场景是频繁对原始数组的某个区间的元素进行增减。
1、首先 构造差分数组 diff ,diff [ i ] = num [ i ] – num [ i – 1 ]

int[] diff = new int[nums.length];
// 构造差分数组
diff[0] = nums[0];
for (int i = 1; i < nums.length; i++) { 
   
    diff[i] = nums[i] - nums[i - 1];
}

在这里插入图片描述
通过这个diff差分数组是可以反推出原始数组nums的,代码逻辑如下:

int[] res = new int[diff.length];
// 根据差分数组构造结果数组
res[0] = diff[0];
for (int i = 1; i < diff.length; i++) { 
   
    res[i] = res[i - 1] + diff[i];
}

2、这样构造差分数组 diff,就可以 快速进行区间增减的操作,如果你想对 区间 [ i , j ] 的元素全部加 3,那么只需要让 diff[ i ] += 3,然后再让 diff[ j + 1 ] -= 3 即可:
在这里插入图片描述
原理很简单,回想 diff 数组反推 nums 数组的过程diff [ i ] += 3 意味着给 nums [ i… ] 所有的元素都加了 3,然后 diff [ j + 1 ] -= 3 又意味着对于 nums [ j + 1… ] 所有元素再减 3,那综合起来,是不是就是对 nums [ i … j ] 中的所有元素都加 3 了 。
只要花费 O(1) 的时间修改 diff 数组,就相当于给 nums 的整个区间做了修改。多次修改diff,然后通过 diff 数组反推,即可得到 nums 修改后的结果。

class Difference { 
   
    // 差分数组
    private int[] diff;
    public Difference(int[] nums) { 
   
        assert nums.length > 0;
        diff = new int[nums.length];
        // 构造差分数组
        diff[0] = nums[0];
        for (int i = 1; i < nums.length; i++) { 
   
            diff[i] = nums[i] - nums[i - 1];
        }
    }
    /* 给闭区间 [i,j] 增加 val(可以是负数)*/
    public void increment(int i, int j, int val) { 
   
        diff[i] += val;
        if (j + 1 < diff.length) { 
   
            diff[j + 1] -= val;
        }
    }
    public int[] result() { 
   
        int[] res = new int[diff.length];
        // 根据差分数组构造结果数组
        res[0] = diff[0];
        for (int i = 1; i < diff.length; i++) { 
   
            res[i] = res[i - 1] + diff[i];
        }
        return res;
    }
}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/134177.html原文链接:https://javaforall.net

(0)
上一篇 2022年6月7日 上午10:00
下一篇 2022年6月7日 上午10:00


相关推荐

  • 贴片电阻电容命名及封装[通俗易懂]

    贴片电阻电容命名及封装[通俗易懂]常见的标准零件件主要有以下几种:电阻(R)、排阻(RA或RN)、电感(L)、陶瓷电容(C)、排容(CP)、钽质电容(C)、二极管(D)、晶体管(Q)。一、零件规格:零件规格即零件的外形尺寸,SMT(表面封装技术)发展至今,业界已经形成了一个标准零件系列,各家零件供货商皆是按这一标准制造。标准零件之尺寸规格有英制与公制两种表示方法,参照下面的常见贴片电阻尺寸表(1inch=25.4m…

    2022年8月21日
    9
  • Hunyuan-MT Pro零基础教程:5分钟搭建专业级翻译网站

    Hunyuan-MT Pro零基础教程:5分钟搭建专业级翻译网站

    2026年3月12日
    2
  • redis 使用

    redis 使用文章目录补充说明语法选项参数实例连接服务端添加数据查询数据删除数据补充说明 yum 安装的 redis conf 在 etc redis redis conf 语法 redis cli 选项 参数 选项 a 输入密码 n 选择数据库若无此参数默认选中 0 数据库参数 set 添加数据 keys 用于查询此参数后可输入正则查询 keys del 删除数据实例连接服务端无密码链接 redis cli 有密码链接 redis cli127 0 0 1 63

    2026年3月17日
    2
  • backtrace调试程序段错误

    backtrace调试程序段错误1 backtrace 的用处一般察看函数运行时堆栈的方法是使用 GDB bt 命令 之类的外部调试器 但是 有些时候为了分析程序的 BUG 主要针对长时间运行程序的分析 在程序出错时打印出函数的调用堆栈是非常有用的 主要用于程序异常退出时寻找错误原因 通常情况下 程序发生段错误时系统会发送 SIGSEGV 信号给程序 缺省处理是退出函数 我们可以使用 signal SIGSEGV amp y

    2026年3月18日
    2
  • Kimi|月之暗面

    Kimi|月之暗面

    2026年3月12日
    1
  • 解决电脑出现 R6034 Runtime Error

    解决电脑出现 R6034 Runtime Error这个问题的出现比较普遍 主要有几个方面 1 病毒引起的 对电脑全盘杀毒 2 硬件上的原因 主要是内存条不兼容引起的 必要时更换内存 3 系统或其它软件引起的 1 系统本身有问题 及时安装官方发行的补丁 必要时重装系统 2 某个软件出现的问题 这里主要是看看开机时运行的软件 用 360 等检查开机运行的软件 把不必要运行的软件都去掉 软件冲突 卸载有问题的软件 下面有两

    2026年3月16日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号