我爱学算法之—— 前缀和(上)

我爱学算法之—— 前缀和(上)

一、【模板】前缀和

题目解析

在这里插入图片描述
这道题,给定一个长度为n的数组,和m次询问;

每一次询问给出两个整数lr,让我们求出区间[l , r]中所有数的和,然后输出。

算法思路

这道题暴力解法:

首先是m次查询(m次测试),每一个给定一个lr,让我们求区间[l , r]中所有数的和。

暴力解法就非常简单了,直接遍历区间[l , r],求出区间中所有数的和即可。

暴力解法时间复杂度O(m * n),也就是O(n^2)级别的时间复杂度;

暴力解法会超时,我们这里想一想可不可以对暴力解法进行一些优化:

  1. 首先m次查询,很显然是不能进行优化的。
  2. 我们只能对求区间[l , r]中所有数的和进行优化。

那如何优化呢?

遍历区间[l , r]来求和时间复杂度是O(n),那我们可不可以用O(1)的复杂度来获得区间[l , r]中所有数的和呢?
在这里插入图片描述

通过上图,我们可以发现:我们要求的[l , r]区间的和s就等于区间[1 , r]的和 减去区间[1 , l]的和。

前缀和

所以,我们可以通过运算来用O(1)的时间复杂度获得区间[l , r]中所有数的和;但是我们要用到区间[1 , l]和区间[1 , r]中所有数的和。

所以我们预先既要处理一个前缀和数组dp:其中dp[i]:表示区间[1 , i]中所有数的和。填写前缀和数组:dp[i] = dp[i-1] + arr[i](也就是前面所有数的和加上当前位置的数)。计算区间[l , r]中所有数的和:dp[r] - dp[l-1](这里区间[l , r]包含l位置,所以要减去dp[i-1]
这里可以说:前缀和和动态规划的大致思路非常相似:

状态表示dp[i]表示区间[1 , i]中所有数的和

状态转移方程dp[i] = dp[i] + arr[i];

获取区间[l , r]中所有数的和s = dp[r] - dp[l-1];

代码实现

#include<cmath>#include<iostream>usingnamespace std;constint N =100001;longlong dp[N];int arr[N];int n, m;intmain(){ cin >> n >> m;for(int i =1; i <= n; i++){ cin >> arr[i]; dp[i]= dp[i -1]+ arr[i];}while(m--){int l, r; cin >> l >> r; cout << dp[r]- dp[l -1]<< endl;}return0;}

二、【模板】二维前缀和

题目解析

在这里插入图片描述
对于这道题,给定一个n*m的二维数组,以及q次查询;

每一次查询给定x1,x2,y1,y2,我们要求以(x1,y1)为左上角,(x2,y2)为右下角的子矩阵中所有数的和。

算法思路

暴力解法:

q次查询,每一查询给定x1,y1,x2,y2,遍历整个子矩阵进行求和操作。

时间复杂度:O(n*m*q),也就是O(n^3)级别的时间复杂度。

很显然会超时,对暴力解法进行优化,很显然只能优化求子矩阵中所有元素的和。

暴力解法中,遍历整个子矩阵去求和,这样太麻烦了;我们可不可以使用O(1)的时间复杂度拿到子矩阵中所有数的和?

当然也是可以的,这就像数学当中求一块面积的和一样。
在这里插入图片描述

如上图所示,我们要求以(x1 , y1)为左上角,(x2 , y2)为右下角的子矩阵中所有数的和,也就是S

我们只要知道s1(以(1 , 1)为左上角,(x1-1 , y1)为右下角的子矩阵的和)、s2(以(1 , 1)为左上角,(x2, y1-1)为右下角的子矩阵的和)、s3(以(1 , 1)为左上角,(x1-1 , y1-1)为右下角的子矩阵的和)以及s4(以(1 , 1)为左上角,(x2 , y2)为右下角的子矩阵的和)。

我们就可以通过数学运算来求Ss = s4 - s1 - s2 + s3

也就是s = dp[x2][y2] - dp[x2][y1-1] - dp[x1-1][y2] + dp[x1-1][y1-1]

这样我们在填写前缀和表时:

在这里插入图片描述

dp[i][j] = dp[i][j-1] +dp[i-1][j] - dp[i-1][j-1] + arr[i][j]

这里也可以将前缀和理解为动态规划

状态表示dp[i][j]表示以(1,1)为左上角,(i,j)为右下角的子矩阵中所有数的和。

状态转移方程dp[i][j] = dp[i][j-1] +dp[i-1][j] - dp[i-1][j-1] + arr[i][j]

计算子矩阵中所有数的和s = dp[x2][y2] - dp[x2][y1-1] - dp[x1-1][y2] + dp[x1-1][y1-1]

代码实现

#include<iostream>usingnamespace std;constint N =1001;int arr[N][N];longlong dp[N][N];int n, m, q;int x1, x2, y1, y2;intmain(){ cin >> n >> m >> q;for(int i =1; i <= n; i++){for(int j =1; j <= m; j++){ cin >> arr[i][j]; dp[i][j]= dp[i -1][j]+ dp[i][j -1]- dp[i -1][j -1]+ arr[i][j];}}while(q--){ cin >> x1 >> y1 >> x2 >> y2; cout <<(dp[x2][y2]- dp[x1 -1][y2]- dp[x2][y1 -1]+ dp[x1 -1][y1 -1])<< endl;}return0;}

总结

这里简单总结一下前缀和算法:

首先前缀和算法可以用来快速的求出子数组/子矩阵中所有数的和,在涉及到求子数组/子矩阵的和时,能够利用前缀和算法来快速的求和。

其次,使用前缀和,我们就要预先构建一个前缀和数组并填写该数组;(和动态规划类似)

注意:在构建前缀和数组时,通常下标从1开始,因为在填写数组时要用到dp[i-1]

最后,前缀和算法就是空间换时间,通过预先构建前缀和数组,让我们能够在O(1)的数据复杂度拿到子数组/子矩阵的和。

到这里本篇文章内容就结束了,感谢各位大佬的支持

Read more

Flutter for OpenHarmony: Flutter 三方库 husky 守卫鸿蒙项目的 Git 提交规范(前端工程化必备)

Flutter for OpenHarmony: Flutter 三方库 husky 守卫鸿蒙项目的 Git 提交规范(前端工程化必备)

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net 前言 在 OpenHarmony 项目的团队协作中,我们最怕遇到“带病提交”的代码。比如:某位开发者提交的代码没经过 dart format 美化、或是包含明显的 lint 警告,甚至导致整个鸿蒙工程编译失败。如果在 CI(持续集成)阶段才发现,修复成本就太高了。 husky 是从前端生态圈引进的 Git Hooks 管理神器。它能让你极简地配置 Git 的各个钩子(如 pre-commit),在代码真正提交到远端(AtomGit)之前,强制执行格式化或单元测试,确保入库的代码永远是高质量的。 一、Git Hook 工作流模型 husky 在本地提交阶段建立了一道自动化的“安检门”。 通过 失败

By Ne0inhk
【汉化中文版】OpenClaw(Clawdbot/Moltbot)第三方开源汉化中文发行版部署全指南:一键脚本/Docker/npm 三模式安装+Ubuntu 环境配置+中文汉化界面适配开源版

【汉化中文版】OpenClaw(Clawdbot/Moltbot)第三方开源汉化中文发行版部署全指南:一键脚本/Docker/npm 三模式安装+Ubuntu 环境配置+中文汉化界面适配开源版

OpenClaw这是什么? OpenClaw(曾用名 Clawdbot / Moltbot)是一个开源的个人 AI 助手平台(GitHub 120k+ Stars),可以通过 WhatsApp、Telegram、Discord 等聊天软件与 AI 交互。简单说就是:在你自己的机器上运行一个 AI 助手,通过常用聊天软件跟它对话。 forks项目仓库 :https://github.com/MaoTouHU/OpenClawChinese 文章目录 * OpenClaw这是什么? * 汉化效果预览 * 环境要求 * 安装方式 * 方式 A:一键脚本(推荐新手) * 方式 B:npm 手动安装 * 方式 C:Docker 部署(服务器推荐) * 首次配置 * 运行初始化向导 * 安装守护进程(

By Ne0inhk

全面的System Verilog教程:从基础到高级验证

本文还有配套的精品资源,点击获取 简介:System Verilog是用于系统级验证、芯片设计与验证以及FPGA实现的强大硬件描述语言。它扩展了Verilog的基础特性,支持高级语言结构,如类、接口、任务和函数,优化了验证流程。教程内容涵盖System Verilog的基础概念、结构化编程元素、并发与同步机制、现代验证方法学、UVM验证方法论以及标准库的应用。旨在教授学生掌握System Verilog语法和高级特性,实现高效、可维护的验证代码。 1. System Verilog概述及应用领域 1.1 System Verilog的起源与发展 System Verilog是作为硬件设计和验证领域的重要语言,由Verilog发展而来,随后被进一步扩展以满足现代电子设计自动化的需要。其发展始于20世纪90年代,目的是在原有Verilog HDL的基础上,提供更为强大的设计验证功能。 1.1.1 Verilog与VHDL的区别 虽然Verilog和VHDL都是硬件描述语言(HDL),但它们在语法和使用方法上存在差异。Verilog更接近于C语言,而VHDL的语法结构则更接近

By Ne0inhk

千元级六轴机械臂DIY完全指南:从零打造工业级开源机器人

千元级六轴机械臂DIY完全指南:从零打造工业级开源机器人 【免费下载链接】Faze4-Robotic-armAll files for 6 axis robot arm with cycloidal gearboxes . 项目地址: https://gitcode.com/gh_mirrors/fa/Faze4-Robotic-arm 还在为工业机械臂动辄数万元的价格望而却步?Faze4开源六轴机械臂项目将彻底改变你的认知!这款采用3D打印技术和创新传动设计的低成本机器人,让每个人都能拥有属于自己的工业级机械臂。无论你是机器人爱好者、学生还是创客,这篇文章将为你提供完整的入门教程和实用指南。 🤔 为什么选择Faze4开源机械臂? 价格革命:传统工业六轴机械臂价格通常在2-5万元,而Faze4通过3D打印技术和开源设计,将成本控制在千元级别,真正实现了"人人可拥有"的机器人梦想。 技术突破:采用创新的3D打印谐波减速器技术,在保证运动精度的同时大幅降低了制造成本。完整的六自由度设计支持复杂轨迹规划,性能媲美商用产品。 学习价值:从硬件搭建到软件开发,全程参与能让你深入理

By Ne0inhk