【算法基础】一维差分 + 二维差分

news2025/1/24 14:54:26

在这里插入图片描述

👦个人主页:Weraphael
✍🏻作者简介:目前正在学习c++和算法
✈️专栏:【C/C++】算法
🐋 希望大家多多支持,咱一起进步!😁
如果文章有啥瑕疵
希望大佬指点一二
如果文章对你有帮助的话
欢迎 评论💬 点赞👍🏻 收藏 📂 加关注😍


回顾

  • 一维前缀和 + 二维前缀和

目录

  • 回顾
  • 一、一维差分
    • 1.1 什么是差分
    • 1.2 如何构造`b`数组
    • 1.3 用途
    • 1.4 代码
      • 模板1
      • 模板2 + 解释
  • 二、二维差分
    • 2.1 什么是二维差分
    • 2.2 如何构建`b[i][j]`以及用途
    • 2.3 代码模板
  • 三、总结

一、一维差分

1.1 什么是差分

  • 首先有一个原数组aa[1],a[2],a[3] ... a[N]
  • 然后构造一个数组bb[1],b[2],b[3]...b[N],使得a[N] = b[1] + b[2]+ b[3] + ... + b[N]
  • 所以,我们就称a数组是b数组的前缀和,而b数组就称为a数组的差分

1.2 如何构造b数组

可以利用高中的知识:

一开始初始化a[0] = 0

b[1] = a[1] - a[0]

b[2] = a[2] - a[1]

b[3] = a[3] - a[2]

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

最后再把上述式子加起来,就可以得到:a[n] = b[1] + b[2]+ b[3] + ... + b[n],所以,构造b[n]数列的公式为:b[n] = a[n] - a[n-1]

1.3 用途

假设现在有一问题:将a数组[l,r]区间中的每一个数都加一常数C,即a[l] + c,a[l+1] + c + ... + a[r] + c(减上一个常数c也是同一个意思)

  • 暴力

for循环遍历[l,r],时间复杂度是O(n)

  • 差分

因为a数组是b数组的前缀和:a[n] = b[1] + b[2] + b[3] + ... + b[n],只要b[l] + c,a数组在区间[l,n]都会加上c,即a[l] + c,a[l + 1] + c,...,a[n] + c,什么意思呢?画个图就豁然开朗了(如下图)
在这里插入图片描述

但是,a数组在区间[l,n]都会加上c,而我们只想在[l,r]上加上c,所以数组b区间[r+1,n]每个数都要-c

在这里插入图片描述

一维差分总结:

  • 给区间[l,r]中的每个数加上cb[l] += cb[r+1] -= c,并且时间复杂度是O(1)
  • 下标为什么从1开始其实和前缀和是一个意思 -> 传送门

1.4 代码

模板1

#include <iostream>
using namespace std;

const int N = 10010;

int a[N],b[N];///全局变量,初始化为0

int main()
{
    int n;// n - 数组元素个数
    scanf("%d",&n);
    //输入原数组a
    for (int i = 1;i <= n;i++) 
        scanf("%d", &a[i]);
        
    //构建b数组
    for (int i = 1;i <= n;i++)
        b[i] = a[i] - a[i - 1];
        
    //a数组在[l,r]上,加上一个数
    int l,r,c;
    scanf("%d%d%d",&l,&r,&c);
    b[l] += c;
    b[r + 1] -= c;
    
    //求出+c后的数组
    for (int i = 1;i <= n;i++)
        b[i] = b[i - 1] + b[i];//a[i] = b[i] + a[i - 1]
    
    //输出
    for(int i = 1;i <= n;i++)
        printf("%d ",b[i]);
    
    return 0;
}

模板2 + 解释

#include <iostream>
using namespace std;

const int N = 10010;

int a[N],b[N];//全局变量,初始化为0

void insert(int l,int r,int c)
{
    b[l] += c;
    b[r + 1] -= c;
}

int main()
{
    int n;
    scanf("%d",&n);
    //输入原数组a
    for (int i = 1;i <= n;i++) 
        scanf("%d", &a[i]);
        
    //构建b数组
    for (int i = 1;i <= n;i++)
        insert(i,i,a[i]);
        
    //a数组在[l,r]上,加上一个数
    int l,r,c;
    scanf("%d%d%d",&l,&r,&c);
    insert(l,r,c);
    
    //求出+c后的数组(前缀和)
    for (int i = 1;i <= n;i++)
        b[i] = b[i] + b[i - 1];//a[i] = b[i] + a[i - 1]
    
    //输出
    for(int i = 1;i <= n;i++)
        printf("%d ",b[i]);
    
    return 0;
}

解释:
首先封装了insert函数来帮助我们在某段区间加上c,而为什么构造b数组时也能使用这个函数呢?这个问题其实我也想了很久,但是其实画个图就明白了

在这里插入图片描述

二、二维差分

2.1 什么是二维差分

  • 首先有一个原数组a[i][j]
  • 然后构建一个差分数组b[i][j],即a[n][m] = b[1][1] + b[2][1] + b[2][2] + ... + b[n][m]
  • 所以,我们称a[i][j]b[i][j]的前缀和数组,而b数组就是a数组的差分

2.2 如何构建b[i][j]以及用途

首先先讲讲用途,假设现在有一问题:已知原数组a中被选中的子矩阵是以(x1,y1)为左上角,以(x2,y2)为右下角所围成的区域,现要求在子矩阵中每个数+c

  • 暴力做法

for循环遍历,时间复杂度是O(n2)

  • 差分

一定要记住,a数组是b数组的前缀和,若对b数组的b[i][j]的修改,势必会影响到a数组中从a[i][j]及往后的每一个数,时间复杂度可以由O(n2)优化成O(1)

做法如下

目的:让黄色区域所有数都加上c

在这里插入图片描述

记住,a数组是b数组的前缀和,若对b数组加上c,势必会影响c。所以b[x1][y1]+c,会导致红色区域加上c

在这里插入图片描述

所以,现只要把绿色部分多加的c去掉即可

在这里插入图片描述

首先,先减去紫色部分多加的c

在这里插入图片描述

  • b[x2 + 1][y1] -= c

接下来再减去灰色部分多加的c

在这里插入图片描述

b[x1][y2 + 1] -= c

最后,由于前两步多减了橙色部分的c,所以还要再加回去

在这里插入图片描述

b[x2 +1 ][y2 + 1] += c

我们把上述过程封装成一个insert函数

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

最后b数组的构造也能使用这个插入函数,过程和一维差分差不多,详情请看一维差分的模板2

2.3 代码模板

#include <iostream>
using namespace std;

const int N = 10010;

int a[N][N],b[N][N];

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

int main()
{
    int n,m;
    scanf("%d%d%d",&n,&m);
    //输入原数组a + 构造差分数组b
    for (int i = 1;i <= n;i++)
    {
        for (int j = 1;j <= m;j++)
        {
            scanf("%d",&a[i][j]);
            insert(i,j,i,j,a[i][j]);
        }
    }
    
    //在指定子矩阵+c
    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");
    }
    return 0;
}

三、总结

一维差分

void insert(int l,int r,int c)
{
    b[l] += c;
    b[r + 1] -= c;
}

二维差分

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/363529.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

#笨鸟先飞 猴博士电路笔记 第一篇 电路基础

第零课 基础知识串联与并联电源电势与电位差第一课 电阻电路的等效变换电压源串联电流源并联电压源和电流源串联电压源和电流源并联电压源转化为电流源电流源转化为电压源Δ-Y等效变换第二课 基尔霍夫定律基尔霍夫电流定律任一结点上流出电流之和等于流入电流之和。受控电流源&…

Java 集合 --- 如何遍历Map

Java 集合 --- 如何遍历MapMap的基本操作如何遍历MapType of HashMapMap没有继承Collection接口AbstractMap和AbstractCollection是平级关系 Map的基本操作 package map; import java.util.*; /*** This program demonstrates the use of a map with key type String and val…

case的使用

1.x和z值 1.1.定义 x&#xff1a;表示不定值 z&#xff1a;表示高阻态&#xff0c;还有一种表达方式“&#xff1f;” 一个x/z可以用来定义十六进制&#xff08;h&#xff09;数的4位二进制的状态&#xff0c;八进制&#xff08;o&#xff09;数的3位&#xff0c;二进制&#x…

「我有一剑可开天门」大厂面试真题,这边建议是直接开冲

前言 说一下&#xff0c;最新在重温雪中悍刀行这本小说&#xff0c;故此有了这么一个沙雕标题&#xff08;小声bb。这本书是真的好看&#xff09;&#xff0c;这套面试题是一个粉丝总结完发给我的&#xff0c;本意是想让我分享出来帮助到更多的人&#xff0c;我整理了一下&…

失眠时还在吃它?有风险,你了解过吗

失眠&#xff0c;是当代人的通病。所以解决失眠也成了刚需&#xff0c;市面上开始出现各种助眠产品。有商业机构调查发现&#xff0c;62%的90后消费者曾买过助眠产品&#xff0c;其中人气选手就是褪黑素。褪黑素本身就是人体天然存在的&#xff0c;与睡眠有关的物质&#xff0c…

第10天-商品服务 - 分层领域模型及规格参数编码实现

1.分层领域模型规约 DO&#xff08; Data Object&#xff09;&#xff1a; 此对象与数据库表结构一一对应&#xff0c;通过 DAO 层向上传输数据源对象。DTO&#xff08; Data Transfer Object&#xff09;&#xff1a;数据传输对象&#xff0c; Service 或 Manager 向外传输的…

VSCode远程连接服务器

工作使用服务器的jupyter&#xff0c;直到有一天服务器挂了&#xff0c;然而&#xff0c;代码还没有来得及备份。o(╥﹏╥)o VScode远程连接服务器&#xff0c;使用服务器的资源&#xff0c;代码可以存在本地&#xff0c;可以解决上述困境。 1.官网下载VSCode.网址https://cod…

JAVA并发集合之ConcurrentHashMap

ConcurrentHashMap是一个支持高并发更新与查询的哈希表(基于HashMap)。Hashmap在多线程并发的情况下&#xff0c;是线程不安全的&#xff0c;容易出现死循环、死锁等问题&#xff0c;JDK8后不会出现死锁问题&#xff0c;但依然存在多线程的系列问题&#xff0c;如&#xff1a;数…

数据结构_ 堆结构与堆排序(c++ 实现 + 完整代码 )

堆结构与堆排序 文章目录堆结构与堆排序引入堆堆结构所满足的数学特性准备代码----------- 往堆中插入元素----------- 删除堆顶堆排序构建完整代码及测试动态分配版本非动态版本引入堆 二叉树 具有左孩子与右孩子的最普通的二叉树。 满二叉树 特殊的二叉树&#xff1a;每个节…

sql join、left join、full join的区别总结,注意事项

1. 结论图 见 https://www.runoob.com/sql/sql-join.html 2. 测试 2.1. 造数据 数据表 mysql脚本 DROP TABLE IF EXISTS class; CREATE TABLE class (c_id INTEGER NOT NULL COMMENT 班级ID,c_name VARCHAR(100) NOT NULL COMMENT 班级名,PRIMARY KEY (c_id) ) CO…

JavaEE简单示例——动态SQL的复杂查询操作<foreach>

简单介绍&#xff1a; 在我们之前学习MySQL的时候&#xff0c;我们曾经有一个操作叫做查询区间&#xff0c;比如我们使用in关键字查询id为3到6之间的值&#xff0c;或者查询id小于100的值&#xff0c;这时候如果将SQL语句一条一条的查询出来进行筛选效率就太慢了&#xff0c;所…

【05-JVM面试专题-运行时数据区的结构都有哪些?哪些是共享的呢?哪些是非共享的呢?详细的介绍一下运行时数据区结构各部分的作用?】

运行时数据区的结构都有哪些&#xff1f;哪些是共享的呢&#xff1f;哪些是非共享的呢&#xff1f;详细的介绍一下运行时数据区结构各部分的作用&#xff1f; 运行时数据区的结构都有哪些&#xff1f;哪些是共享的呢&#xff1f;哪些是非共享的呢&#xff1f;详细的介绍一下运行…

带您了解TiDB MySQL数据库中关于日期、时间的坑

带您了解TiDB & MySQL数据库中关于日期、时间的坑时间的基础知识什么是时间计算时间的几种方法世界时&#xff08;UT&#xff09;协调世界时&#xff08;UTC&#xff09;国际原子时&#xff08;TAI&#xff09;时区的概念中国所在的时区操作系统的时区datetimedatectl数据库…

Spring代理模式——静态代理和动态代理

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…

python打包exe实用工具auto-py-to-exe的操作方法

auto-py-to-exe操作方法auto-py-to-exe 是一个用于打包 python 程序的程序。本文就是主要介绍如何使用 auto-py-to-exe 完成 python 程序打包。本文主要分为两节&#xff0c;第一节主要对 auto-py-to-exe 做一些介绍&#xff0c;第二节则是演示 auto-py-to-exe 的打包过程。一、…

pygraphviz安装教程

0x01. 背景 最近在做casual inference&#xff0c;做实验时候想因果图可视化&#xff0c;遂需要安装pygraphviz&#xff0c;整了一下午&#xff0c;终于捣鼓好了&#xff0c;真头大。 环境&#xff1a; win10操作系统python3.9环境 0x02. 安装Graphviz 传送门&#xff1a;…

linux:本地套接字通信客户和服务器代码

客户端代码 #include <stdlib.h> #include <stdio.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <string.h> #include <arpa/inet.h> #include <sys/un.h> int main(int argc, const cha…

中间件安全—Tomcat常见漏洞

中间件安全—Tomcat常见漏洞1.Tomcat常见漏洞1.1.前言1.2.文件上传 (CVE-2017-12615)1.2.1.漏洞原理1.2.2.影响版本1.2.3.漏洞复现1.2.3.1.测试是否允许PUT1.2.3.2.验证漏洞是否存在1.2.3.3.访问test.jsp1.2.3.4.上传执行命令脚本1.2.3.5.执行命令1.3.文件包含漏洞&#xff08;…

【第一章 - 绪论】- 数据结构(近八千字详解)

目录 一、 数据结构的研究内容 二、基本概念和术语 2.1 - 数据、数据元素、数据项和数据对象 2.2 - 数据结构 2.2.1 - 逻辑结构 2.2.2 - 存储结构 2.3 - 数据类型和抽象数据类型 三、抽象数据类型的表现与实现 四、算法和算法分析 4.1 - 算法的定义及特性 4.2 - 评价…

手把手教大家在 gRPC 中使用 JWT 完成身份校验

文章目录1. JWT 介绍1.1 无状态登录1.1.1 什么是有状态1.1.2 什么是无状态1.2 如何实现无状态1.3 JWT1.3.1 简介1.3.2 JWT数据格式1.3.3 JWT 交互流程1.3.4 JWT 存在的问题2. 实践2.1 项目创建2.2 grpc_api2.3 grpc_server2.4 grpc_client3. 小结上篇文章松哥和小伙伴们聊了在 …