【algorithm】认真讲解前缀和与差分 (图文搭配)

这篇具有很好参考价值的文章主要介绍了【algorithm】认真讲解前缀和与差分 (图文搭配)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

🚀write in front🚀
📝个人主页:认真写博客的夏目浅石.
📣系列专栏:AcWing算法笔记

今天的月色好美
前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言


前言

这里介绍以下前缀和算法以及差分算法,用来梳理自己所学到的算法知识。


一、前缀和算法

1.1 什么是前缀和?

从我的理解角度来讲:前缀和就是高中数学当中的数列的求和Sn,差分就是前缀和的逆运算,就是递推公式。

1.2 一维前缀和

先来看一道题目吧:
前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言
这是之前训练的时候的一道经典的前缀和问题,我们很容易想到暴力作法:遍历数组

代码如下:

#include<stdio.h>
const int N = 1e5 + 10;
int a[N];
int n,m;
int main()
{
	scanf("%d%d", &n, &m);
	for(int i = 1; i <= n; i++) scanf("%d", &a[i]);
	while(m--)
	{
    	int l, r;
    	int sum = 0;
    	scanf("%d%d", &l, &r);
    	for(int i = l; i <= r; i++)
    	{ 
     	   sum += a[i];
  	  	}
    	printf("%d\n",sum);
	}
	return 0;
}

这样的时间复杂度为O(n * m),如果n和m的数据量稍微大一点就有可能超时,而我们如果使用前缀和的方法来做的话就能够将时间复杂度降到O(n + m),大大提高了运算效率。

前缀和做法:

#include<stdio.h>
int main()
{
	long long n,k,arr[100010],sum[100010];
	scanf("%lld %lld",&n,&k);
	sum[0]=0;
	for(int i=1;i<=n;i++)
	{
		scanf("%lld",&arr[i]);
		int tmp=arr[i];
		sum[i]=tmp+sum[i-1];
	}
	for(int i=1;i<=k;i++)
	{
		int f,t;
		scanf("%d %d",&f,&t);
		printf("%lld\n",sum[t]-sum[f-1]);//重要步骤
	}
	return 0;
}

原理讲解:

sum[r] = a[1] + a[2] + a[3] + a[l-1] + a[l] + a[l + 1] .. a[r];
sum[l - 1] = a[1] + a[2] + a[3] + ... + a[l - 1];
sum[r] - sum[l - 1] = a[l] + a[l + 1] + ... + a[r];

这样,对于每个询问,只需要执行 sum[r] - sum[l - 1]。输出原序列中从第l个数到第r个数的和的时间复杂度变成了O(1)。

我们把它叫做一维前缀和。

二、二维前缀和

先来看一道题目吧:

前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言

因为这里提及到了二维这个词,所以我们先来定义一个二维数组s[][] , s[i][j] 表示二维数组中,左上角(1, 1)到右下角(i, j)所包围的矩阵元素的和。接下来推导二维前缀和的公式。

先看一张图:
前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言
图解:

  • (1,1)(i,j-1)表示的面积是S1+S2定义为S黄蓝
  • (1,1)(i-1,j)表示的面积是S1+S3定义为S黄粉
  • (1,1)(i,j)表示的面积是S黄蓝+S黄粉-S1+S4

因此得出二维前缀和预处理公式

s[i][j] = s[i - 1][j] + s[i][j - 1 ] + a[i] [j] - s[i - 1][j - 1]

讲解完这些基础知识就可以去解决刚才的问题啦

#include <iostream>
using namespace std;
const int N = 1010;
int n, m, q;
int s[N][N];
int main()
{
    scanf("%d%d%d", &n, &m, &q);
    for (int i = 1; i <= n; i ++ )
        for (int j = 1; j <= m; j ++ )
            scanf("%d", &s[i][j]);
    for (int i = 1; i <= n; i ++ )
        for (int j = 1; j <= m; j ++ )
            s[i][j] += s[i - 1][j] + s[i][j - 1] - s[i - 1][j - 1];
    while (q -- )
    {
        int x1, y1, x2, y2;
        scanf("%d%d%d%d", &x1, &y1, &x2, &y2);
        printf("%d\n", s[x2][y2] - s[x1 - 1][y2] - s[x2][y1 - 1] + s[x1 - 1][y1 - 1]);
    }
    return 0;
}

所以总结模板就是:

S[i, j] = 第i行j列格子左上部分所有元素的和
以(x1, y1)为左上角,(x2, y2)为右下角的子矩阵的和为:
S[x2, y2] - S[x1 - 1, y2] - S[x2, y1 - 1] + S[x1 - 1, y1 - 1]

三、一维差分

先看一道问题:
前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言

类似于数学中的求导和积分,差分可以看成前缀和的逆运算。
首先给定一个原数组aa[1], a[2] , , , , a[n];

然后我们构造一个数组b b[1], b[2] , , , b[i];

使得 a[i] = b[1] + b[2] + , , , + b[i]

也就是说,a数组是b数组的前缀和数组,反过来我们把b数组叫做a数组的差分数组。换句话说,每一个a[i]都是b数组中从头开始的一段区间和。

其实换个好理解的方式:
a[0 ]= 0;
b[1] = a[1] - a[0];
b[2] = a[2] - a[1];

b[n] = a[n] - a[n - 1];

但是知道了这些怎么用到题目上呢?或者换句话说,怎么就成为一种算法了呢?hhh下面就来解决这个问题哦~

如果给定区间[l, r ],让我们把a数组中的[l, r] 区间中的每一个数都加上c,即 a[l] + c , a[l + 1] + c , a[l + 2] + c ,,,,,, a[r] + c;

暴力做法是for循环l到r区间,时间复杂度O(n),如果我们需要对原数组执行m次这样的操作,时间复杂度就会变成O(n * m)。有没有更高效的做法吗? 考虑差分的做法。

首先让差分b数组中的 b[l] + c ,通过前缀和运算,a数组变成 a[l] + c ,a[l + 1] + c,,,,,, a[n] + c;

然后我们打个补丁,b[r + 1] - c, 通过前缀和运算,a数组变成 a[r + 1] - c,a[r + 2] - c,,,,,,,a[n] - c;

前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言
b[l] + c,效果使得a数组中 a[l] 及以后的数都加上了c(红色部分),但我们只要求l到r 区间加上 c, 因此还需要执行 b[r + 1] - c,让a数组中 a[r + 1]及往后的区间再减去c(绿色部分),这样对于a[r] 以后区间的数相当于没有发生改变。

因此我们得出一维差分结论:给a数组中的[ l, r] 区间中的每一个数都加上c,只需对差分数组bb[l] + = c,b[r+1] - = c 。时间复杂度为O(1), 大大提高了效率。

代码如下:

#include<stdio.h>
int main()
{
    int arr[100010],a[100010],n,m,q;
    scanf("%d%d%d",&n,&m);
    for(int i=1;i<=n;i++) scanf("%d",&arr[i]);
    a[0]=0;
    for(int i=1;i<=n;i++)
    {
        a[i]=arr[i]-arr[i-1];
    }
    while(m--)
    {
        int l,r,c;
        scanf("%d%d%d",&l,&r,&c);
        a[l]+=c;
        a[r+1]-=c;
        
    }
    for(int i=1;i<=n;i++)
    {
        arr[i]=arr[i-1]+a[i];
        printf("%d ",arr[i]);
    }
    
    return 0;
}

四、二维差分

首先先看一道题目:
前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言
如果扩展到二维,我们需要让二维数组被选中的子矩阵中的每个元素的值加上c,是否也可以达到O(1)的时间复杂度。答案是可以的,考虑二维差分。

那么下面就来讲解二维差分

前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言

  • b[x1][y1] += c ; 对应图1 ,让整个a数组中黄色矩形面积的元素都加上了c。
  • b[x1,][y2 + 1] -= c ; 对应图2 ,让整个a数组中粉色+绿色矩形面积的元素再减去c,使其内元素不发生改变。
  • b[x2 + 1][y1] -= c ; 对应图3 ,让整个a数组中蓝色+绿色矩形面积的元素再减去c,使其内元素不发生改变。
  • b[x2 + 1][y2 + 1] += c; 对应图4,让整个a数组中绿色矩形面积的元素再加上c,绿色内的相当于被减了两次,再加上一次c,才能使其恢复。

模板:

void insert(int x1,int y1,int x2,int y2,int c)
{
    b[x1][y1]+=c;
    b[x1][y2+1]-=c;
    b[x2+1][y1]-=c;
    b[x2+1][y2+1]+=c;
}

前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言

b[i][j] = a[i][j] − a[i − 1][j] − a[i][j − 1] + a[i −1 ][j − 1]
代码如下:

#include<iostream>
#include<cstdio>
using namespace std;

const int N = 1e3 + 10;
int a[N][N],b[N][N];
int n,m,q;

void insert(int x1,int y1,int x2,int y2,int c)
{
    b[x1][y1]+=c;
    b[x1][y2+1]-=c;
    b[x2+1][y1]-=c;
    b[x2+1][y2+1]+=c;
}

int main()
{
    cin>>n>>m>>q;
    
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=m;j++)
        {
            scanf("%d",&a[i][j]);
        }
    }
    
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=m;j++)
        {
            insert(i,j,i,j,a[i][j]);//构造差分数组
        }
    }
    
    while(q--)
    {
        int x1,y1,x2,y2,c;
        cin>>x1>>y1>>x2>>y2>>c;
        insert(x1,y1,x2,y2,c);//进行差分
    }
    
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=m;j++)
        {
            b[i][j]+=b[i-1][j]+b[i][j-1]-b[i-1][j-1];
        }
    }
    
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=m;j++)
        {
            printf("%d ",b[i][j]);
        }
        printf("\n");
    }

}

总结

今天学习了前缀和算法知识,每天进步一点点,不积硅步,无以至千里
我们下期见吧~

前缀和 和差分 应用场景,AcWing算法笔记,算法,c语言,c++,数据结构,开发语言
如果无聊的话,就来逛逛我的博客栈吧stack-frame.cn

原创不易,还希望各位大佬支持一下 \textcolor{blue}{原创不易,还希望各位大佬支持一下} 原创不易,还希望各位大佬支持一下

👍 点赞,你的认可是我创作的动力! \textcolor{9c81c1}{点赞,你的认可是我创作的动力!} 点赞,你的认可是我创作的动力!

⭐️ 收藏,你的青睐是我努力的方向! \textcolor{ed7976}{收藏,你的青睐是我努力的方向!} 收藏,你的青睐是我努力的方向!

✏️ 评论,你的意见是我进步的财富! \textcolor{98c091}{评论,你的意见是我进步的财富!} 评论,你的意见是我进步的财富!
文章来源地址https://www.toymoban.com/news/detail-811422.html

到了这里,关于【algorithm】认真讲解前缀和与差分 (图文搭配)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【OI学习笔记】基础算法-前缀和与差分算法

    板块:基础算法、线性优化 难度:较易 前置知识:C++基础语法 在一维空间中,对于一个数据总量为 n n n 的数组 a a a ,有数据 a [ 1 ] , a [ 2 ] , a [ 3 ] , . . . , a [ n − 1 ] , a [ n ] a[1],a[2],a[3],...,a[n-1],a[n] a [ 1 ] , a [ 2 ] , a [ 3 ] , ... , a [ n − 1 ] , a [ n ] ,定义一个数组 s u m sum s u m ,

    2024年02月08日
    浏览(32)
  • 【有营养的算法笔记】基础算法 —— 推导证明前缀和与差分

    👑作者主页:@安 度 因 🏠学习社区:StackFrame 📖专栏链接:有营养的算法笔记 如果无聊的话,就来逛逛 我的博客栈 吧! 🌹 Hello,小伙伴们,好几天没有更新了,今天更了一篇比较“硬核的文章”。 主要内容为前缀和与差分算法的推导证明和代码实现。这篇文章博主还是画

    2024年01月21日
    浏览(37)
  • [Algorithm][前缀和][和为K的子数组][和可被K整除的子数组][连续数组][矩阵区域和]详细讲解

    和为 K 的子数组 分析 :因为有 负数 的存在 无法使用\\\" 双指针 \\\"优化,因为 区间求和不具备单调性 可能已经找到前缀和为k的子数组了,但是后面仍然可能会有更长的和为k的子数组存在 思路 :前缀和 + 哈希表 前缀和 : 以 i 位置为结尾 的所有的子数组 将问题转化为 :在

    2024年04月28日
    浏览(25)
  • C++ 二维差分 二维前缀和逆运算 差分矩阵

    输入一个 n 行 m 列的整数矩阵,再输入 q 个操作,每个操作包含五个整数 x1,y1,x2,y2,c ,其中 (x1,y1) 和 (x2,y2) 表示一个子矩阵的左上角坐标和右下角坐标。 每个操作都要将选中的子矩阵中的每个元素的值加上 c 。 请你将进行完所有操作后的矩阵输出。 输入格式 第一行包含整数

    2024年02月21日
    浏览(42)
  • 高精度/前缀和/差分

    存储方式: 整数的长度一般小于1e6 大整数的每一位存储到数组里 存储时低位在前,高位在后,方便进位 高精度加法 每一位相加Ai + Bi + t, t表示进位取值0/1,逢十进一 模板: 高精度减法 每一位相减Ai - Bi - t, t 表示借位取值0/1 模板: 高精度乘法 A * b ,b=10000, len(A) = 1e6 , 乘的

    2024年02月16日
    浏览(30)
  • 前缀和&差分

    前缀和:一段序列里的前n项和 给出n个数,在给出q次问询,每次问询给出L、R,快速求出每组数组中一段L至R区间的和 给出一段数组,每次问询为求出l到r区间的和 普通方法: L到R进行遍历,那么在每次求区间和的过程中时间复杂度为O(n),q次问询时间复杂度为O(q*n) 前缀和:

    2024年02月15日
    浏览(25)
  • 前缀和 差分

    对于数列A,它的前缀和数列S[i]就表示数列A从第一个元素到第i个元素的总和。 前缀和的主要用处:求任意区间的区间和 一般通过遍历求和的时间复杂度是O(n),通过前缀和可以减少为O(1) 具体解法如下: ​前缀和计算区间[l,r]的区间和:S[r] - S[l - 1] ACWing 795前缀和 Tips: ​让元素

    2024年02月05日
    浏览(41)
  • 前缀和、差分

    前缀和可以快速求区间和。 差分相当于前缀和的逆运算。 前缀和、差分都是以空间换时间的算法 注意:本文图文并茂 将提供以下图文链接供大家理解: 图文链接: 飞书图解链接🎉🎉🎉 密码: 672Z172 定义 前缀和可以简单理解为「数列的前 n 项的和」,是一种重要的预处

    2024年02月05日
    浏览(34)
  • 算法基础之差分和前缀和

    结论:差分是前缀和的逆运算 举例 一维差分 二维差分 用在一维子区间里的数都增减一个固定值,把对于区间内每个数的操作转换为对于差分数组中的端点的操作,时间复杂度降为o(1)。 用在二维子矩阵里的数都增减一个固定值,把对于子矩阵的每个数的操作转换为对应差分

    2024年02月07日
    浏览(31)
  • 第四章:前缀和、差分(数列)

    在解释什么是前缀和之前,我们先回顾一下高中学过的数列: 我们这里所说的前缀和其实就是我们在高中学的数列中的 Sn(前n项和) ,只是我们这里需要将 S1 , S2 , S3 , S4 …… Sn 当作一个新的数组。 为了这个式子的高度统一性,我们的S0和a0都是不存储数据的,将其设置为0,这

    2023年04月08日
    浏览(22)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包