如何在 Java 中利用数组实现简单的前缀和(Prefix Sum)技术以支持 O(1) 时间的区间查询
如何在 Ja va 中利用数组实现简单的前缀和(Prefix Sum)技术以支持 O(1) 时间的区间查询

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈
在 Ja va 开发中,尤其是处理数据查询时,你是否遇到过这样的场景:需要频繁计算一个数组中某个连续区间的元素之和?如果每次都去遍历累加,效率显然不高。这时候,前缀和(Prefix Sum)技术就该登场了。它的核心思路非常巧妙:通过一次性的预处理,将后续所有区间求和的复杂度降至 O(1)。
构造前缀和数组
具体怎么实现呢?关键在于构建一个辅助数组。假设我们有一个原始数组 nums,长度为 n。通常,我们会选择构建一个长度为 n + 1 的“带哨兵”前缀和数组 prefix。这个小小的“+1”设计,能让我们在后续计算时省去不少边界判断的麻烦。
- 首先,初始化
prefix[0] = 0。 - 然后,从
i = 1开始遍历到n,执行prefix[i] = prefix[i - 1] + nums[i - 1]。
这样一来,prefix[i] 存储的值,就恰好是原数组 nums 中从第 0 个元素到第 i-1 个元素的总和。整个预处理过程只需要 O(n) 的时间。
实现 O(1) 区间查询
预处理完成后,真正的魔法就发生了。当我们需要查询原数组中任意闭区间 [left, right](其中 0 ≤ left ≤ right < n)的和时,只需要一个简单的减法:
sum = prefix[right + 1] - prefix[left]
这个公式为什么成立?我们拆解一下:
prefix[right + 1]代表了nums[0]到nums[right]的所有元素之和。prefix[left]代表了nums[0]到nums[left - 1]的所有元素之和。- 两者相减,中间重叠的部分被抵消,剩下的正是我们需要的
nums[left]到nums[right]的区间和。
看,一次查询,无论区间多长,都只需要常数时间,效率的提升立竿见影。
Ja va 示例代码
理论说清楚了,来看一个完整、可直接运行的 Ja va 示例代码,它会让你理解得更透彻:
public class PrefixSum { private int[] prefix;public PrefixSum(int[] nums) { int n = nums.length; prefix = new int[n + 1]; for (int i = 1; i <= n; i++) { prefix[i] = prefix[i - 1] + nums[i - 1]; } } // 查询 [left, right] 闭区间的和,O(1) public int rangeSum(int left, int right) { return prefix[right + 1] - prefix[left]; } // 使用示例 public static void main(String[] args) { int[] nums = {2, 1, 3, 5, 4}; PrefixSum ps = new PrefixSum(nums); System.out.println(ps.rangeSum(1, 3)); // nums[1]+nums[2]+nums[3] = 1+3+5 = 9 System.out.println(ps.rangeSum(0, 4)); // 全数组和 = 15 }}
注意事项与扩展
当然,前缀和技术并非万能钥匙。它最适合处理“静态数组”,也就是那些初始化后就不太会频繁修改元素值的场景。如果业务需求中需要支持大量的单点更新操作,那么前缀和每次更新都需要重新计算一大片,效率就不够看了。这时候,就该考虑更高级的数据结构,比如线段树或者树状数组(Fenwick Tree)。
最后,再划几个重点:
- 复杂度:空间复杂度为 O(n),预处理时间复杂度为 O(n),查询则是 O(1)。
- 边界检查:在实际生产代码中,务必记得在查询方法中加入索引越界的校验,例如判断
left和right是否在合法范围内,以及left是否小于等于right。- 思路延伸:这个一维前缀和的思路完全可以推广到二维,也就是“二维前缀和”,用于高效计算一个矩阵中任意子矩阵内所有元素的和,这在图像处理、动态规划等算法中非常有用。
相关攻略
Sublime Text 能不能当主力 Ja va IDE? 答案很明确:不能。但它完全可以胜任一个高效的“轻量级开发终端”。关键在于认清它的能力边界——Sublime Text 本身并不具备 Ja va 语法解析、类路径管理、Ma ven 生命周期控制,或是连接 JVM 调试协议(JDWP)的能力
基于位运算的容差检测报告优化方案 在工业级数据校验场景中,比如木材尺寸的容差检测,我们常常需要根据多个布尔状态(如厚度、宽度、长度是否合格)来组合生成差异化的提示信息。传统的实现方式,往往是写下一长串的 if-else 分支,来覆盖所有可能的逻辑组合。功能虽然能实现,但问题也很明显:代码重复度高,扩
如何在 Ja va 中使用 ArrayList ensureCapacity() 减少由于频繁增删导致的数组重分配 ensureCapacity() 真的能减少重分配吗? 答案是肯定的,但这里有个关键前提:它只对“新增”操作有效,而且必须在执行大量 add() 之前就调用。至于 remove() 操
如何在 Ja va 中利用 while 循环实现一个简单的基于时间轮算法的定时任务调度流程 可行但仅适用于学习、嵌入式或教学场景;生产环境应优先选用 HashedWheelTimer、ScheduledThreadPoolExecutor 或 Quartz。 在 Ja va 中,用 while
如何在 Ja va 中使用 String matches() 编写带有“零宽断言”的高级正则校验表达式 说起 Ja va 里的 String matches() 方法,很多开发者都踩过同一个坑:它要求正则表达式必须从头到尾、完完整整地匹配整个字符串。这相当于在模式前后自动加上了 ^ 和 $。所以,当
热门专题
热门推荐
企业介绍信写作指南:掌握这份正式商务“名片”的核心要素与实用范文 在商业合作与行政事务中,企业介绍信是一份至关重要的正式文书。它不仅是身份与意图的权威证明,更是建立信任、开启合作的“通行证”。一份撰写规范、信息完备的介绍信,能有效提升沟通效率,保障业务顺畅推进。本文将深入解析企业介绍信的写作要点,并
学校实习介绍信模板大全 在现代职场与高校人才培养体系中,实习介绍信已成为连接校园与社会的重要桥梁。作为一份具有正式效力的官方推荐文书,它不仅为学生开启实践之门,也为用人单位提供了可靠的背景参考。为帮助广大师生高效处理实习事务,我们精心整理并优化了以下几款高实用性的学校实习介绍信标准模板,供您直接套用
每到新环境,一份得体的自我介绍往往是开启人际交往的第一扇门。下面这份“2026年新生入学自我介绍”灵感合集,旨在为即将步入新阶段的你提供实用参考与创意启发。 2026年新生入学自我介绍【一】 尊敬的老师,亲爱的同学们: 大家好。关于“懂事”这个词,我记忆中最深刻的一次体验,发生在我四岁那年。 那时,
近期,BLUR币因其在NFT市场的活跃表现备受关注 最近,NFT交易平台币BLUR在圈内的讨论度明显升温。它本质上是一个专注于NFT交易和社区生态的平台代币,核心目标很明确:提升NFT市场的交易效率和用户体验,同时通过一套精心设计的激励机制,把更多的玩家和收藏家吸引到这个生态里来。 对于刚接触这个领
2026级大学新生自我介绍范文【一】 大家好,我是来自XX高中的XX。如果学科也有性格,我想我与文学最为投契。相较于理科世界中严谨的公式与抽象的几何,文学世界里流淌的人文气息与思想深度,总能更深地触动我的内心。在独处的时光里,与一本好书为伴是最惬意的事。沉浸于经典著作所构建的广阔世界,品味字里行间浓





