0

0

c++中如何实现前缀和_c++数组前缀和算法实现技巧【汇总】

穿越時空

穿越時空

发布时间:2026-01-17 16:14:02

|

268人浏览过

|

来源于php中文网

原创

前缀和是手动构建的数组技巧,可将区间求和从O(n)降至O(1),但要求数组不频繁修改;一维需构建n+1长prefix数组,定义prefix[i]为arr[0..i-1]和,查询[l,r]用prefix[r+1]-prefix[l];二维用dpi表示左上矩形和,构建时dpi=dpi-1+dpi-dpi-1+matrixi-1,查询(r1,c1)到(r2,c2)用dpr2+1-dpr1-dpr2+1+dpr1;std::partial_sum仅适用于简单一维场景,不支持二维或自定义运算;若需频繁单点更新,应改用树状数组或线段树。

c++中如何实现前缀和_c++数组前缀和算法实现技巧【汇总】

前缀和在 C++ 中不是语言内置功能,而是靠手动构建的数组技巧;它能将区间求和从 O(n) 降为 O(1),但必须满足「数组不频繁修改」的前提。

一、一维数组前缀和:从 vector 构建 prefix 数组

核心是定义 prefix[i] 表示原数组 arr[0..i-1] 的和(即左闭右开),这样 arr[l..r] 的和就是 prefix[r+1] - prefix[l],避免边界特判。

常见错误包括:下标越界、用 prefix[i] = prefix[i-1] + arr[i] 导致 prefix 和原数组对齐(易错配区间)。

  • 推荐初始化方式:prefix[0] = 0,长度为 n+1
  • 递推式必须是:prefix[i+1] = prefix[i] + arr[i]
  • 查询 [l, r](含端点):写成 prefix[r+1] - prefix[l],别漏 +1
vector arr = {1, 2, 3, 4, 5};
vector prefix(arr.size() + 1, 0);
for (int i = 0; i < arr.size(); ++i) {
    prefix[i + 1] = prefix[i] + arr[i];
}
// 查询 [1, 3] → arr[1]+arr[2]+arr[3] = 2+3+4 = 9
int sum = prefix[4] - prefix[1]; // prefix[3+1] - prefix[1]

二、二维数组前缀和:用 dp[i][j] 表示左上角矩形和

二维前缀和本质是容斥:以 (i,j) 为右下角的矩形和 = 左 + 上 - 左上 + 当前值。同样建议多开一行一列,让 dp[0][*]dp[*][0] 全为 0。

立即学习C++免费学习笔记(深入)”;

容易踩的坑是公式记混,或在查询时没统一坐标系(比如把输入的行列当成 0-index 却按 1-index 写公式)。

医真AI+开放平台
医真AI+开放平台

医真AI+ 医学AI开放平台

下载
  • 构建时:确保 ij 都从 1 开始遍历,dp[i][j] = dp[i-1][j] + dp[i][j-1] - dp[i-1][j-1] + matrix[i-1][j-1]
  • 查子矩阵 (r1,c1)(r2,c2)(含端点,0-index 输入):用 dp[r2+1][c2+1] - dp[r1][c2+1] - dp[r2+1][c1] + dp[r1][c1]
  • 若用 vector>,注意初始化大小:行数 = 原行数+1,列数 = 原列数+1

三、前缀和 vs std::partial_sum:何时用标准库

std::partial_sum 是 C++ 标准算法,可直接生成前缀和,但它只做“就地累加”或输出到另一容器,不解决二维、不支持自定义运算(如异或前缀和)、也不提供查询封装。

它适合快速原型或简单一维场景,但工程中常需自己封装类(如 PrefixSum1D)来绑定数据、防误用、支持 update(配合树状数组)等。

  • 使用示例:partial_sum(arr.begin(), arr.end(), prefix.begin() + 1);(假设 prefix[0]==0
  • 注意:它不会自动多开一位,要自己保证目标容器足够大
  • 不能替代二维逻辑——标准库无 partial_sum_2d

四、修改频繁?别硬套前缀和,考虑 std::vector + 树状数组或线段树

原始前缀和一旦构建,单点修改代价是 O(n)(得重算后面所有)。如果题目有「单点更新 + 区间查询」需求,前缀和就不适用了。

这时候应切换数据结构:树状数组(BIT)代码短、常数小;线段树更通用但稍重。二者都支持 O(log n) 更新与查询。

  • 判断信号:题干出现 “update index x to value y” 或 “dynamic array” 字样
  • 别试图在前缀和数组上做懒更新——没有意义,复杂度不降
  • 树状数组实现只需 10 行左右,比手写带修改的前缀和更可靠

前缀和真正省事的地方,是它足够简单——几行循环 + 一次减法就能替代嵌套循环。但它的脆弱性也在这里:只要数据动了,整个加速逻辑就崩了。用之前,先盯住「改不改」这三个字。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

317

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

534

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

16

2026.01.06

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

402

2023.08.14

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 6.9万人学习

C 教程
C 教程

共75课时 | 4.1万人学习

C++教程
C++教程

共115课时 | 12.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号