C++ 前缀和数组
一. 一维数组前缀和
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;
}
相关文章:

C++ 前缀和数组
一. 一维数组前缀和 1.1. 定义 前缀和算法通过预处理数组,计算从起始位置到每个位置的和,生成一个新的数组(前缀和数组)。利用该数组,可以快速计算任意区间的和,快速求出数组中某一段连续区间的和。 1.2. …...
PHP 实现通用数组字段过滤函数:灵活去除或保留指定 Key
PHP 实现数组去除或保留指定字段的通用函数详解 一、文章标题 《PHP 实现通用数组字段过滤函数:灵活去除或保留指定 Key》 二、摘要 在实际开发中,我们经常需要对数组进行字段级别的操作,例如从一个数组中删除某些敏感字段(如密码、token),或者只保留特定字段用于接口…...
NACOS2.3.0开启鉴权登录
环境 名称版本nacos2.3.0(Linux)java java version "17.0.14" 2025-01-21 LTS # # Copyright 1999-2021 Alibaba Group Holding Ltd. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use thi…...

细胞冻存的注意事项,细胞冻存试剂有哪些品牌推荐
细胞冻存的原理 细胞冻存的基本原理是利用低温环境抑制细胞的新陈代谢,使细胞进入一种“休眠”状态。在低温条件下,细胞的生物活动几乎停止,从而实现长期保存。然而,细胞在冷冻过程中可能会因为细胞内外水分结冰形成冰晶而受损。…...

快速上手Linux火墙管理
实验网络环境: 主机IP网络f1192.168.42.129/24NATf2(双网卡) 192.168.42.128/24 192.168.127.20/24 NAT HOST-NOLY f3192.168.127.30/24HOST-ONLY 一、iptables服务 1.启用iptables服务 2.语法格式及常用参数 语法格式:参数&…...

[创业之路-375]:企业战略管理案例分析 - 华为科技巨擘的崛起:重构全球数字化底座的超级生命体
在人类文明从工业时代(机械、电气、自动化)迈向数字智能(硬件、软件、算法、虚拟、智能)时代的临界点上,一家中国企业正以令人震撼的姿态重塑全球科技版图。从通信网络的底层架构到智能终端的生态闭环,从芯…...
【paddle】常见的数学运算
根据提供的 PaddlePaddle 函数列表,我们可以将它们按照数学运算、逻辑运算、三角函数、特殊函数、统计函数、张量操作和其他操作等类型进行分类。以下是根据函数功能进行的分类: 取整运算 Rounding functions 代码描述round(x)距离 x 最近的整数floor(…...

AI基础知识(05):模型提示词、核心设计、高阶应用、效果增强
目录 一、核心设计原则 二、高阶应用场景 三、突破性技巧 以下是针对DeepSeek模型的提示词设计思路及典型应用场景示例,帮助挖掘其潜在能力: 一、核心设计原则 1. 需求明确化:用「角色定位任务目标输出格式」明确边界 例:作为历…...
分布式事务之Seata
概述 Seata有四种模式 AT模式:无侵入式的分布式事务解决方案,适合不希望对业务进行改造的场景,但由于需要添加全局事务锁,对影响高并发系统的性能。该模式主要关注多DB访问的数据一致性,也包括多服务下的多DB数据访问…...

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

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

写实交互数字人在AI招聘中的应用方案
随着科技的进步,越来越多的行业开始探索如何利用人工智能提升效率和服务质量。其中,写实交互数字人技术以其高度拟真的交互体验和丰富的情感表达能力,在人力资源领域特别是招聘环节中展现出了巨大潜力。本文将探讨写实交互数字人在AI招聘中的…...
C++中IO类(iostream、fstream和sstream)知识详解和应用
一、C I/O 类体系概览 C 的 I/O 功能由一组 流(stream) 类封装,位于头文件 <iostream>、<fstream>、<sstream> 等。核心类别及其继承关系简图如下: ios_base↑basic_ios<CharT,Traits>↑┌───────…...
Spring Boot中如何对密码等敏感信息进行脱敏处理
以下是常见的脱敏方法及实现步骤,涵盖配置、日志和API响应等多个层面: 1. 配置文件敏感信息脱敏 (1) 使用加密库(如Jasypt) 步骤: 添加依赖: <dependency><groupId>com.github.ulise…...
React从基础入门到高级实战:React 基础入门 - JSX与组件基础
JSX 与组件基础 引言 在 React 开发中,JSX 和 组件 是两个最基础且核心的概念。JSX 是一种独特的语法,让你在 JavaScript 中编写类似 HTML 的代码,而组件则是 React 应用的基本构建块,帮助你将复杂的界面拆分为可复用的模块。本…...

房贷利率计算前端小程序
利率计算前端小程序 视图效果展示如下: 在这里插入代码片 <!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,那么我可以使用2022的Visual Studio。 查看Toolkit版本 nvcc -V 配置 ok,让我们开始Visual Studio的nvcc编译器配置 参考例文https://github.com/apachecn/succinc…...

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

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 无索引,无序,不可重复 HashSet object类中默认hashCode的方法是根据地址值。 如果集合中存储的是自定义对象,必须要重写hashCode和equals方法。 底层原理 jdk8以前:数组 链表 jdk8及以后:数组 链表 红黑…...
Elasticsearch简单集成java框架方式。
Elasticsearch 在 Java 中最常用的客户端是什么?如何初始化一个 RestHighLevelClient?如何用 Spring Boot 快速集成 Elasticsearch?Spring Data Elasticsearch 如何定义实体类与索引的映射? 最常用的 Java 客户端 目前官方推荐使用…...
【RAG文档切割】从基础拆分到语义分块实战指南
目录 🌟 前言🏗️ 技术背景与价值🩹 当前技术痛点🛠️ 解决方案概述👥 目标读者说明 🧠 一、技术原理剖析📊 分块流程架构图💡 核心分块策略🔧 关键技术模块 Ὦ…...
stream数据流
核心知识点:数据流(Stream Data Flow) 1. 通俗易懂的解释 想象一下你正在用花园里的水管浇花。水管里的水不是一次性全部倒出来的,而是持续不断地从水龙头流出,经过水管,最终从喷头喷洒到花上。在这个过程…...
利用 XML 外部实体注入(XXE)读取文件和探测内部网络
利用 XML 外部实体注入(XXE)读取文件和探测内部网络 引言 XML 外部实体注入(XXE)是一种常见的安全漏洞,攻击者可以通过这种漏洞读取服务器上的文件或探测内部网络。本文将通过一个实际的 Python 代码示例,…...

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

Visual Studio 2019/2022:当前不会命中断点,还没有为该文档加载任何符号。
1、打开调试的模块窗口,该窗口一定要在调试状态下才会显示。 vs2019打开调试的模块窗口 2、Visual Studio 2019提示未使用调试信息生成二进制文件 未使用调试信息生成二进制文件 3、然后到debug目录下看下确实未生成CoreCms.Net.Web.WebApi.pdb文件。 那下面的…...

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

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

【MySQL】CRUD
CRUD 简介 CRUD是对数据库中的记录进行基本的增删改查操作 Create(创建)Retrieve(读取)Update(更新)Delete(删除) 一、新增(Create) 语法: I…...

Spring Boot微服务架构(三):Spring Initializr创建CRM项目
使用Spring Initializr创建CRM项目 一、创建项目前的准备 访问Spring Initializr网站: 打开浏览器访问 https://start.spring.io/或者直接使用IDE(如IntelliJ IDEA或Eclipse)内置的Spring Initializr功能 项目基本信息配置: Proj…...