【入门级-算法-4、算法策略:差分】

一、概念

差分是一种处理区间修改和单点查询的高效技巧。它通过构建一个辅助数组(差分数组),将原数组的区间操作转化为对差分数组的单点操作,从而将时间复杂度从 O (n) 降至 O (1)。

二、差分的核心原理

差分通过构建一个辅助数组(差分数组)来简化原数组的区间更新。

首先进行差分数组的定义

给定原数组 a1...n,其差分数组 d1...n 定义为:

d1 = a1

d2 = a2 - a1

...

di = ai - ai-1(其中 i > 1)

原数组与差分数组的关系:差分数组的前缀和等于原数组。

d1 = a1

d2 = a2 - a1

d3 = a3 - a2

d4 = a4 - a3

...

也就是说,对任意 i,ai = d1 + d2 + ... + di

这是差分能够反向推导原数组的关键。

三、差分的核心应用:区间更新

差分的最大价值在于高效处理"对原数组某一区间 l, r 内所有元素加 / 减一个值 k" 的操作。

区间更新的转化规则无需遍历原数组的 l, r 区间,只需对差分数组 d 做两处单点修改:

dl = dl+ k:表示从 l 位置开始,原数组所有元素增加 k。

dr+1= ar+1- ar

由于此时ar已经增加了k,而 ar+1没有变化,因此新的dr+1减少了k

即新的dr+1=旧的dr+1- k(若 r+1 <= n):表示从 r+1 位置开始,抵消之前的增加操作,确保仅 l, r 区间受影响。

操作流程示例

假设原数组 a = 1, 2, 3, 4, 5,需对 2,4 区间各元素加 2:

初始差分数组 d = 1, 1, 1, 1, 1

执行更新:d2 += 2 → d2 = 3;d5 -= 2 → d5 = -1

更新后差分数组 d = 1, 3, 1, 1, -1

求前缀和得到新原数组:a = 1, 4, 5, 6, 5,符合预期。

四、差分的适用场景

差分并非万能,它仅适用于特定类型的问题,主要包括:

多组区间更新 + 一次最终查询:例如多次给不同区间加值,最后输出完整数组。

线性结构数据:仅适用于一维数组或可转化为一维数组的场景(如二维差分需特殊处理)。

五、代码实现

题目:给定数组 a,多次执行:对 l, r 内所有数 +val,最后输出整个数组。

c 复制代码
#include <stdio.h>
#define N 100

int main() {
    int a[N] = {0};  // 原数组
    int d[N] = {0};  // 差分数组
    int n, m;
    int i;

    // 1. 输入数组长度 n
    printf("请输入元素个数 n:");
    scanf("%d", &n);

    printf("请输入 %d 个整数:", n);
    for (i = 1; i <= n; i++) {
        scanf("%d", &a[i]);
    }

    // 2. 初始化差分数组
    d[1] = a[1];
    for (i = 2; i <= n; i++) {
        d[i] = a[i] - a[i - 1];
    }

    // 3. 操作次数 m
    printf("请输入操作次数 m:");
    scanf("%d", &m);

    // 4. 执行 m 次区间加操作
    while (m--) {
        int l, r, val;
        printf("输入 l r val:");
        scanf("%d %d %d", &l, &r, &val);

        d[l] += val;
        if (r + 1 <= n)
            d[r + 1] -= val;
    }

    // 5. 求前缀和,还原数组
    a[1] = d[1];
    for (i = 2; i <= n; i++) {
        a[i] = a[i - 1] + d[i];
    }

    // 6. 输出结果
    printf("最终数组:");
    for (i = 1; i <= n; i++) {
        printf("%d ", a[i]);
    }
    printf("\n");

    return 0;
}

运行程序

输入:

请输入元素个数 n:5

请输入 5 个整数:1 2 3 4 5

请输入操作次数 m:2

输入 l r val:2 4 10

输入 l r val:1 3 5

过程:

原数组:1 2 3 4 5

操作 1:2,4 +10 → 1 12 13 14 5

操作 2:1,3 +5 → 6 17 18 14 5

输出结果:

最终数组:6 17 18 14 5

相关推荐
kkeeper~15 小时前
0基础C语言积跬步之数据在内存中的存储
c语言·数据结构·算法
2401_8685347815 小时前
论企业网络设计
数据结构
wabs66616 小时前
关于贪心算法的一些自我总结【力扣45.跳跃游戏II】【灵感来源:代码随想录】
算法·贪心算法·复盘
2401_8769641316 小时前
【湖北专升本】2026湖北专升本真题PDF+备考资料汇总
数据结构·人工智能·经验分享·深度学习·算法·计算机视觉
嗝o゚17 小时前
CANN GE 算子融合——融合算法与调度策略
算法·昇腾·cann·ge
小江的记录本17 小时前
【JVM虚拟机】垃圾回收GC:垃圾回收算法:标记-清除、标记-复制、标记-整理、分代收集(附《思维导图》+《面试高频考点清单》)
java·jvm·后端·python·算法·安全·面试
Ulyanov18 小时前
用声明式语法重新定义Python桌面UI:QML+PySide6现代开发入门(一)
开发语言·python·算法·ui·系统仿真·雷达电子对抗仿真
数据科学小丫18 小时前
特征工程处理
人工智能·算法·机器学习
z落落19 小时前
C#参数区别
java·算法·c#
c2385620 小时前
vector(下)
数据结构·算法