C++ 前缀和数组

news2025/5/25 21:50:09

一.  一维数组前缀和

 1.1. 定义

       前缀和算法通过预处理数组,计算从起始位置到每个位置的和,生成一个新的数组(前缀和数组)。利用该数组,可以快速计算任意区间的和,快速求出数组中某一段连续区间的和。

 

1.2. 公式

1.2.1. 前缀和数组公式

 

prefix[i] 表示的是从数组 nums 的第 0 个元素到第 i - 1 个元素的所有元素之和,也就是前缀和。

prefix[i - 1] 是从第 0 个元素到第 i - 2 个元素的和,

nums[i - 1] 是数组 nums 中第 i - 1 个元素的值。

        所以这个公式的意思是:当前位置的前缀和等于前一个位置的前缀和加上当前位置对应的数组元素的值。

 

假设有一个数组 nums = [1, 2, 3, 4, 5],我们来计算它的前缀和数组。

初始化 prefix[0] = 0(通常会把前缀和数组的第 0 个元素初始化为 0,方便后续计算)。

当 i = 1 时,prefix[1] = prefix[0] + nums[0] = 0 + 1 = 1。

当 i = 2 时,prefix[2] = prefix[1] + nums[1] = 1 + 2 = 3。

当 i = 3 时,prefix[3] = prefix[2] + nums[2] = 3 + 3 = 6。

当 i = 4 时,prefix[4] = prefix[3] + nums[3] = 6 + 4 = 10。

当 i = 5 时,prefix[5] = prefix[4] + nums[4] = 10 + 5 = 15。

 

代码实现:

#include <bits/stdc++.h>

using namespace std;

const int MAXN=1005; //默认数组最大存储空间 

int nums[MAXN]; //实际存储数据的数组,初始化元素值为0

int prefix[MAXN]={0}; //前缀和数组,初始化元素值为0 

int n; //数组实际大小 

int main() {

        cin>>n; //输入数组的大小

        //循环输入数组中的每个元素的值

        for(int i=0;i<n;i++) cin>>nums[i];

        cout<<"输出原数组的值:" ;

        for(int i=0;i<n;i++) cout<<nums[i]<<" "; 

        //构建前缀和数组

        for(int i=1;i<=n;i++){

        prefix[i]=prefix[i-1]+nums[i-1];

 } 

 cout<<"\n输出前缀和的值:" ;

//循环输出前缀和数组中存储的值

for(int i=1;i<=n;i++){

        cout<<prefix[i]<<" ";

 } 

return 0;

}

 

1.2.2、区间和数组公式

sum(i, j) 表示的是数组 nums 中从第 i 个元素到第 j 个元素的所有元素之和。prefix[j + 1] 是从第 0 个元素到第 j 个元素的和。

prefix[i] 是从第 0 个元素到第 i - 1 个元素的和。

那么用 prefix[j + 1] 减去 prefix[i] 就得到了从第 i 个元素到第 j 个元素的和。

还是使用上面的数组 nums = [1, 2, 3, 4, 5],假设我们要计算从第 1 个元素(索引为 0)到第 3 个元素(索引为 2)的和,即 sum(0, 2)。

我们已经计算出前缀和数组 prefix = [0, 1, 3, 6, 10, 15]。

根据公式 sum(0, 2) = prefix[2 + 1] - prefix[0] 

                     = prefix[3] - prefix[0] 

                     = 6 - 0 

                     = 6

 

代码实现:

#include <bits/stdc++.h>

using namespace std;

const int MAXN=1005; //默认数组最大存储空间 

int nums[MAXN]; //实际存储数据的数组,初始化元素值为0

int prefix[MAXN]={0}; //前缀和数组,初始化元素值为0 

int n,l,r; //数组实际大小 l代表起始位置、r代表结束位置 

int main() {

        cin>>n; //输入数组的大小

       //循环输入数组中的每个元素的值

       for(int i=0;i<n;i++) cin>>nums[i];

       cout<<"输出原数组的值:" ;

       for(int i=0;i<n;i++) cout<<nums[i]<<" "; 

       for(int i=1;i<=n;i++){ //构建前缀和数组

               prefix[i]=prefix[i-1]+nums[i-1];

        } 

       cout<<"\n输出前缀和的值:" ;

       for(int i=1;i<=n;i++){//循环输出前缀和数组中存储的值

        cout<<prefix[i]<<" ";

 } 

 cin>>l>>r; //输入计算区间的起始位置

 cout<<prefix[r+1]-prefix[l];

 return 0;

}

 

1.2.3、动态规划思想

        这里的 dp 通常是动态规划数组,dp[i] 表示的是与数组 arr 前 i 个元素相关的某种状态值。这个公式表示当前状态 dp[i] 可以由前一个状态 dp[i - 1] 加上当前数组元素 arr[i] 得到。

        假设我们要计算数组 arr = [1, 2, 3, 4, 5] 中每个位置的累积和,就可以使用这个动态规划的思想。

初始化 dp[0] = arr[0]。

当 i = 1 时,dp[1] = dp[0] + arr[1] = 1 + 2 = 3。

当 i = 2 时,dp[2] = dp[1] + arr[2] = 3 + 3 = 6。

当 i = 3 时,dp[3] = dp[2] + arr[3] = 6 + 4 = 10。

当 i = 4 时,dp[4] = dp[3] + arr[4] = 10 + 5 = 15。

 

1.3、适用题目

区间和查询:如求子数组和。

频率统计:如统计满足条件的子数组数量。

优化时间复杂度:将区间和查询从O(n)降至O(1)。

 

1.4、优缺点
1、优点

高效查询:区间和查询时间复杂度为O(1)。

预处理简单:前缀和数组的构建时间复杂度为O(n)。

2、缺点

空间开销:需要额外O(n)空间存储前缀和数组。

仅适用于静态数组:数组元素不变时适用,频繁更新时不适用。

 

二.  二维数组前缀和

2.1、定义

        二维数组前缀和是一种用于高效处理二维数组区域和查询的数据结构和算法。

        对于一个二维数组arr,其前缀和数组prefix中prefix[i][j]表示从arr[0][0]到arr[i][j]这个矩形区域内所有元素的和。

 

2.2、公式

        构建二维数组前缀和的核心思想是利用动态规划的思想,通过已知的前缀和来计算新的前缀和。

 

2.3、区域和查询

如果我们想查询从 (x1, y1) 到 (x2, y2) 这个矩形区域内所有元素的和,可以使用以下公式:

sum=prefix[x2][y2]−prefix[x1−1][y2]−prefix[x2][y1−1]+prefix[x1−1][y1−1]

        这个公式的含义是:整个大矩形的前缀和减去上方小矩形的前缀和和左方小矩形的前缀和,再加上左上方小矩形的前缀和(因为这个区域被多减了一次)。

 

2.4、代码实现

#include <bits/stdc++.h>

using namespace std;

const int MAXN = 1005; // 默认二维数组最大行数和列数

int nums[MAXN][MAXN] = {0}; // 实际存储二维数据的数组,初始化元素值为 0

int prefix[MAXN][MAXN] = {0}; // 二维前缀和数组,初始化元素值为 0 

int n, m, q; // n 二维数组实际行数、m 二维数组实际列数、q 查询区间的次数

int x,y,x2,y2; // x,y 代表查询区间的左上角坐标,x2,y2 代表查询区间的右下角坐标

 

代码实现:

int main() {

        cin >> n >> m >> q; // 输入二维数组的行数、列数和查询次数

        // 循环输入二维数组中的每个元素的值

        for (int i = 1; i <= n; i++) {

                for (int j = 1; j <= m; j++) {

                cin >> nums[i][j];

         }

}

 // 构建二维前缀和数组

 for (int i = 1; i <= n; i++) {

         for (int j = 1; j <= m; j++) {

         prefix[i][j] = prefix[i - 1][j] + prefix[i][j - 1] - prefix[i - 1][j - 1] + nums[i][j];

         }

}

while (q--) {

         cin >> x >> y >> x2 >> y2; // 输入查询区间的左上角和右下角坐标

        // 计算并输出查询区间的和

        cout << prefix[x2][y2] - prefix[x - 1][y2] - prefix[x2][y - 1] + prefix[x - 1][y - 1] << endl;

    }

    return 0;

}    

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

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

相关文章

细胞冻存的注意事项,细胞冻存试剂有哪些品牌推荐

细胞冻存的原理 细胞冻存的基本原理是利用低温环境抑制细胞的新陈代谢&#xff0c;使细胞进入一种“休眠”状态。在低温条件下&#xff0c;细胞的生物活动几乎停止&#xff0c;从而实现长期保存。然而&#xff0c;细胞在冷冻过程中可能会因为细胞内外水分结冰形成冰晶而受损。…

快速上手Linux火墙管理

实验网络环境&#xff1a; 主机IP网络f1192.168.42.129/24NATf2&#xff08;双网卡&#xff09; 192.168.42.128/24 192.168.127.20/24 NAT HOST-NOLY f3192.168.127.30/24HOST-ONLY 一、iptables服务 1.启用iptables服务 2.语法格式及常用参数 语法格式&#xff1a;参数&…

[创业之路-375]:企业战略管理案例分析 - 华为科技巨擘的崛起:重构全球数字化底座的超级生命体

在人类文明从工业时代&#xff08;机械、电气、自动化&#xff09;迈向数字智能&#xff08;硬件、软件、算法、虚拟、智能&#xff09;时代的临界点上&#xff0c;一家中国企业正以令人震撼的姿态重塑全球科技版图。从通信网络的底层架构到智能终端的生态闭环&#xff0c;从芯…

AI基础知识(05):模型提示词、核心设计、高阶应用、效果增强

目录 一、核心设计原则 二、高阶应用场景 三、突破性技巧 以下是针对DeepSeek模型的提示词设计思路及典型应用场景示例&#xff0c;帮助挖掘其潜在能力&#xff1a; 一、核心设计原则 1. 需求明确化&#xff1a;用「角色定位任务目标输出格式」明确边界 例&#xff1a;作为历…

推测解码算法在 MTT GPU 的应用实践

前言​ 目前主流的大模型自回归解码每一步都只生成一个token, 尽管kv cache等技术可以提升解码的效率&#xff0c;但是单个样本的解码速度依然受限于访存瓶颈&#xff0c;即模型需要频繁从内存中读取和写入数据&#xff0c;此时GPU的利用率有限。为了解决这种问题&#xff0c;…

Axure酒店管理系统原型

酒店管理系统通常被设计为包含多个模块或界面&#xff0c;以支持酒店运营的不同方面和参与者。其中&#xff0c;管理端和商户端是两个核心组成部分&#xff0c;它们各自承担着不同的职责和功能。 软件版本&#xff1a;Axure RP 9 预览地址&#xff1a;https://556i1e.axshare.…

写实交互数字人在AI招聘中的应用方案

随着科技的进步&#xff0c;越来越多的行业开始探索如何利用人工智能提升效率和服务质量。其中&#xff0c;写实交互数字人技术以其高度拟真的交互体验和丰富的情感表达能力&#xff0c;在人力资源领域特别是招聘环节中展现出了巨大潜力。本文将探讨写实交互数字人在AI招聘中的…

房贷利率计算前端小程序

利率计算前端小程序 视图效果展示如下&#xff1a; 在这里插入代码片 <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0&qu…

在Visual Studio中进行cuda编程

首先下载与CUDA Toolkit匹配的Visual Studio版本 比如我的CUDA Toolkit版本是12.6&#xff0c;那么我可以使用2022的Visual Studio。 查看Toolkit版本 nvcc -V 配置 ok&#xff0c;让我们开始Visual Studio的nvcc编译器配置 参考例文https://github.com/apachecn/succinc…

Fastrace:Rust 中分布式追踪的现代化方案

原文链接&#xff1a;Fastrace: A Modern Approach to Distributed Tracing in Rust | FastLabs / Blog 摘要 在微服务架构中&#xff0c;分布式追踪对于理解应用程序的行为至关重要。虽然 tokio-rs/tracing 在 Rust 中被广泛使用&#xff0c;但它存在一些显著的挑战&#xf…

Linux云计算训练营笔记day13【CentOS 7 find、vim、vimdiff、ping、wget、curl、RPM、YUM】

Linux云计算训练营笔记day13[CentOS 7 find、vim、vimdiff、ping、wget、curl、RPM、YUM]] 目录 Linux云计算训练营笔记day13[CentOS 7 find、vim、vimdiff、ping、wget、curl、RPM、YUM]]1.find练习2.vim高级使用2.1 命令模式:2.2 插入模式:2.3 末行模式: 3. vimdiff4. ping5.…

黑马Java基础笔记-15

Set 无索引&#xff0c;无序&#xff0c;不可重复 HashSet object类中默认hashCode的方法是根据地址值。 如果集合中存储的是自定义对象&#xff0c;必须要重写hashCode和equals方法。 底层原理 jdk8以前&#xff1a;数组 链表 jdk8及以后&#xff1a;数组 链表 红黑…

软件设计师“排序算法”真题考点分析——求三连

一、考点分值占比与趋势分析 综合知识题分值统计表 年份考题数量总分值分值占比考察重点2018222.67%时间复杂度/稳定性判断2019334.00%算法特性对比分析2020222.67%空间复杂度要求2021111.33%算法稳定性判断2022334.00%综合特性应用2023222.67%时间复杂度计算2024222.67%分治…

Visual Studio 2019/2022:当前不会命中断点,还没有为该文档加载任何符号。

1、打开调试的模块窗口&#xff0c;该窗口一定要在调试状态下才会显示。 vs2019打开调试的模块窗口 2、Visual Studio 2019提示未使用调试信息生成二进制文件 未使用调试信息生成二进制文件 3、然后到debug目录下看下确实未生成CoreCms.Net.Web.WebApi.pdb文件。 那下面的…

vue--ofd/pdf预览实现

背景 实现预览ofd/pdf超链接功能 业务实现 pdf的预览 实现方式&#xff1a; 直接使用 <iframe :src"${url}#navpanes0&toolbar0" /> 实现pdf的预览。 navpanes0 隐藏侧边栏toolbar0 隐藏顶部工具栏 使用pdf.js&#xff0c;代码先行&#xff1a; <tem…

Python 爬虫之requests 模块的应用

requests 是用 python 语言编写的一个开源的HTTP库&#xff0c;可以通过 requests 库编写 python 代码发送网络请求&#xff0c;其简单易用&#xff0c;是编写爬虫程序时必知必会的一个模块。 requests 模块的作用 发送网络请求&#xff0c;获取响应数据。 中文文档&#xf…

【MySQL】CRUD

CRUD 简介 CRUD是对数据库中的记录进行基本的增删改查操作 Create&#xff08;创建&#xff09;Retrieve&#xff08;读取&#xff09;Update&#xff08;更新&#xff09;Delete&#xff08;删除&#xff09; 一、新增&#xff08;Create&#xff09; 语法&#xff1a; I…

Spring Boot微服务架构(三):Spring Initializr创建CRM项目

使用Spring Initializr创建CRM项目 一、创建项目前的准备 访问Spring Initializr网站&#xff1a; 打开浏览器访问 https://start.spring.io/或者直接使用IDE&#xff08;如IntelliJ IDEA或Eclipse&#xff09;内置的Spring Initializr功能 项目基本信息配置&#xff1a; Proj…

【笔记】PyCharm 中创建Poetry解释器

#工作记录 在使用 PyCharm 进行 Python 项目开发时&#xff0c;为项目配置合适的 Python 解释器至关重要。Poetry 作为一款强大的依赖管理和打包工具&#xff0c;能帮助我们更便捷地管理项目的依赖项与虚拟环境。下面将详细记录在 PyCharm 中创建 Poetry 解释器的步骤。 前提条…

python中的numpy(数组)

&#xff08;0&#xff09;numpy介绍 NumPy是Python中用于科学计算的基础库&#xff0c;提供高效的多维数组对象ndarray&#xff0c;支持向量化运算&#xff0c;能大幅提高数值计算效率。它集成了大量数学函数&#xff08;如线性代数、傅里叶变换等&#xff09;&#xff0c;可…