【数据结构与算法】常用算法 前缀和

news/2024/7/24 6:17:55 标签: 算法, 前缀和, 数据结构

 🎉🎉欢迎光临🎉🎉

🏅我是苏泽,一位对技术充满热情的探索者和分享者。🚀🚀

🌟特别推荐给大家我的最新专栏数据结构算法:初学者入门指南》📘📘

本专栏纯属为爱发电永久免费!!!

这是苏泽的个人主页可以看到我其他的内容哦👇👇

努力的苏泽http://suzee.blog.csdn.net

281d83d3df1a4544bdd00fdadf50950b.png

 

 

引言:


前缀和算法就是一种常用的算法,用于快速计算数组或序列中某一区间的和。它在很多问题中都有广泛的应用,例如求解子数组的最大和、区间和等。本文将介绍前缀和算法的原理及其应用。

  1. 问题背景:
    在很多计算问题中,我们需要频繁地计算数组或序列中某一区间的和。如果每次都遍历区间中的元素进行累加,时间复杂度会很高,效率低下。因此,我们需要一种更高效的方法来解决这个问题。

  2. 前缀和算法的作用:
    前缀和算法可以将数组或序列中的每个位置的元素累加起来,得到一个前缀和数组。利用前缀和数组,我们可以在O(1)的时间复杂度内计算任意区间的和,从而提高计算效率。

2.1 前缀和的定义:


对于一个给定的数组或序列,我们定义前缀和数组prefixSum,其中prefixSum[i]表示原数组中前i个元素的和。即prefixSum[i] = nums[0] + nums[1] + ... + nums[i-1]。特别地,prefixSum[0] = 0。

2.2 计算前缀和数组:


为了计算前缀和数组,我们可以从第一个元素开始遍历原数组,对于每个位置i,将前i个元素的和保存到prefixSum[i]中。具体的计算方法如下:

prefixSum[0] = 0
for i from 1 to n:
    prefixSum[i] = prefixSum[i-1] + nums[i-1]

其中,n表示数组的长度,nums表示原数组。

2.3 利用前缀和数组计算区间和:


利用前缀和数组,我们可以在O(1)的时间复杂度内计算任意区间的和。对于给定的区间[left, right],其中left表示区间的起始位置,right表示区间的结束位置,区间和可以通过前缀和数组计算得到:

sum[left, right] = prefixSum[right+1] - prefixSum[left]

其中,prefixSum[right+1]表示前right+1个元素的和,prefixSum[left]表示前left个元素的和。通过这个计算公式,我们可以快速得到任意区间的和。

3.1 子数组和的计算

def prefix_sum(nums):
    n = len(nums)
    prefix = [0] * (n + 1)
    
    for i in range(1, n + 1):
        prefix[i] = prefix[i - 1] + nums[i - 1]
    
    return prefix

def subarray_sum(nums, start, end):
    prefix = prefix_sum(nums)
    return prefix[end + 1] - prefix[start]

使用 prefix_sum 函数可以计算原始数组的前缀和数组 prefix。然后,通过计算 prefix[end + 1] - prefix[start],就可以得到从下标 start 到下标 end 的子数组和。

3.2 区间和的查询

def range_sum(prefix, start, end):
    return prefix[end + 1] - prefix[start]

在区间和的查询中,我们不需要每次都重新计算前缀和数组。我们可以直接使用已经计算好的前缀和数组 prefix,通过计算 prefix[end + 1] - prefix[start],即可得到从下标 start 到下标 end 的区间和。

3.3 数组元素的更新和查询

def update(prefix, index, value):
    prefix[index + 1] += value

def query(prefix, index):
    return prefix[index + 1]

在数组元素的更新和查询中,我们可以直接通过修改前缀和数组 prefix 中的相应位置来实现。通过 prefix[index + 1] 可以查询下标 index 处的数组元素值,通过 prefix[index + 1] += value 可以更新下标 index 处的数组元素值。

4.1 前缀和数组的空间优化

def prefix_sum(nums):
    n = len(nums)
    prefix = [0] * n
    prefix[0] = nums[0]
    
    for i in range(1, n):
        prefix[i] = prefix[i - 1] + nums[i]
    
    return prefix

def subarray_sum(nums, start, end):
    prefix = prefix_sum(nums)
    if start == 0:
        return prefix[end]
    else:
        return prefix[end] - prefix[start - 1]

前缀和数组的空间优化中,我们可以直接在原始数组 nums 上进行操作,而不需要额外的前缀和数组。在计算子数组和时,通过 prefix[end] - prefix[start - 1] 可以得到从下标 start 到下标 end 的子数组和,特别要注意边界情况。

4.2 优化区间和查询的时间复杂度

def prefix_sum(nums):
    n = len(nums)
    prefix = [0] * (n + 1)
    
    for i in range(1, n + 1):
        prefix[i] = prefix[i - 1] + nums[i - 1]
    
    return prefix

def range_sum(prefix, start, end):
    return prefix[end + 1] - prefix[start]

def optimize_range_sum(prefix, start, end):
    return range_sum(prefix, start, end) if start == 0 else range_sum(prefix, start, end) - prefix[start]

在优化区间和查询的时间复杂度时,我们可以直接利用已经计算好的前缀和数组 prefix,通过 prefix[end + 1] - prefix[start] 可以得到从下标 start 到下标 end 的区间和。如果起始下标 start 不为0,我们可以通过 range_sum(prefix, start, end) - prefix[start] 计算区间和,避免重复计算前缀和

4.3 前缀和数组的差分运算

def difference(nums):
    n = len(nums)
    dif = [0] * n
```python
def difference(nums):
    n = len(nums)
    dif = [0] * n
    dif[0] = nums[0]
    
    for i in range(1, n):
        dif[i] = nums[i] - nums[i - 1]
    
    return dif

def update(dif, start, end, value):
    dif[start] += value
    if end + 1 < len(dif):
        dif[end + 1] -= value

def recover(nums, dif):
    n = len(nums)
    nums[0] = dif[0]
    
    for i in range(1, n):
        nums[i] = dif[i] + nums[i - 1]
    
    return nums

 

通过差分运算,我们可以将原始数组转化为差分数组 dif。差分数组的特点是,差分数组中的值表示原始数组中相邻元素的差值。通过 dif[i] = nums[i] - nums[i - 1],可以得到差分数组。在进行数组元素的更新时,我们只需要更新差分数组中的相应位置,而不需要修改原始数组。通过 update 函数可以更新差分数组的指定范围内的值。最后,通过 recover 函数可以根据差分数组恢复原始数组的值。

5.1 问题描述

假设给定一个整数数组 nums,我们需要找到该数组中连续子数组的最大和。

5.2 基本解法

def max_subarray_sum(nums):
    n = len(nums)
    max_sum = float('-inf')
    
    for i in range(n):
        current_sum = 0
        for j in range(i, n):
            current_sum += nums[j]
            max_sum = max(max_sum, current_sum)
    
    return max_sum

基本解法是使用双重循环来枚举所有可能的连续子数组,并计算它们的和。在计算过程中,记录最大的和,最后返回最大和。

5.3 利用前缀和算法的优化解法

def max_subarray_sum(nums):
    n = len(nums)
    prefix = [0] * (n + 1)
    max_sum = float('-inf')
    
    for i in range(1, n + 1):
        prefix[i] = prefix[i - 1] + nums[i - 1]
    
    for i in range(n):
        for j in range(i, n):
            current_sum = prefix[j + 1] - prefix[i]
            max_sum = max(max_sum, current_sum)
    
    return max_sum

优化解法利用了前缀和算法来减少计算子数组和的时间复杂度。首先,通过计算前缀和数组 prefix,可以在常数时间内得到任意区间的和。然后,使用双重循环来枚举所有可能的连续子数组,但是在计算过程中,通过前缀和数组直接计算子数组的和,而不需要每次都重新计算。最后返回最大和。

总结

6.1 前缀和算法的优点

  • 前缀和算法能够高效地计算数组或序列中某个区间的和,大大减少了重复计算的时间复杂度。
  • 前缀和算法适用于需要频繁查询区间和或数组元素的更新和查询的场景。
  • 前缀和算法可以通过差分运算来进一步优化,减少空间复杂度。

6.2 注意事项和适用范围

  • 在使用前缀和算法时,需要注意边界情况和数组下标的处理,确保计算的正确性。
  • 前缀和算法适用于静态数组,即数组元素在查询和更新操作之间不会发生改变的情况。
  • 如果数组元素会发生频繁的更新操作,需要使用差分运算来处理,以避免重复计算前缀和

 

 


http://www.niftyadmin.cn/n/5395092.html

相关文章

11 文件管理(1)

1.文件的定义 文件可以被解释为一组带标识的、在逻辑上有完整意义的信息项的序列。这个标识为文件名&#xff0c;信息项构成了文件内容的基本单位。信息项是构成文件内容的基本单位&#xff0c;这些信息项是一组有序序列&#xff0c;它们之间具有一定的顺序关系&#xff1b;文件…

MWC 2024丨美格智能发布全新5G-A模组及FWA解决方案,将5.5G带入现实

2月26日&#xff0c;在MWC 2024世界移动通信大会上&#xff0c;美格智能正式宣布推出5G-A模组SRM817WE以及全新的5G-A FWA解决方案&#xff0c;包含5G-A CPE解决方案SRT858M、5G-A MiFi解决方案SRT878H和5G-A ODU解决方案SRT853MX&#xff0c;旨在进一步提升网络性能&#xff0…

2023 re:Invent 用 Amazon Q 打造你的知识库

前言 随着 ChatGPT 的问世&#xff0c;我们迎来了许多创新和变革的机会。一年一度的亚马逊云科技大会 re:Invent 也带来了许多前言的技术&#xff0c;其中 Amazon CEO Adam Selipsky 在 2023 re:Invent 大会中介绍 Amazon Q 让我印象深刻&#xff0c;这预示着生成式 AI 的又一…

【Vue3】学习computed计算属性

&#x1f497;&#x1f497;&#x1f497;欢迎来到我的博客&#xff0c;你将找到有关如何使用技术解决问题的文章&#xff0c;也会找到某个技术的学习路线。无论你是何种职业&#xff0c;我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章&#xff0c;也欢…

【Spring连载】使用Spring Data的Repositories----定义Repository接口

【Spring连载】使用Spring Data的Repositories----定义Repository接口 一、微调Repository定义二、使用多个Spring Data模块的Repositories 要定义repository接口&#xff0c;首先需要定义特定于域&#xff08;domain&#xff09;类的repository接口。接口必须继承Repository&a…

2024年华为OD机试真题-文本统计分析-Python-OD统一考试(C卷)

题目描述: 有一个文件, 包含以一定规则写作的文本, 请统计文件中包含的文本数量 规则如下 1. 文本以";"分隔,最后一条可以没有";",但空文本不能算语句,比如"COMMAND A; ;"只能算一条语句. 注意, 无字符/空白字符/制表符都算作"空&qu…

智能充电桩案例分析——交流充电桩

随着电动汽车的发展&#xff0c;充电桩也成为当下的一个很热门的工业产品。我们初步接触充电桩&#xff0c;有了点滴的感受。 先简单说说容易一点的交流充电桩。就是通过市电&#xff08;220V,50赫兹&#xff09;给电动汽车提供充电的能源来源。很容易理解&#xff0c;交流…

lv21 QT 常用控件 2

1 QT GUI 类继承简介 布局管理器 输出控件 输入控件 按钮 容器 2 按钮示例 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QCheckBox> #include <QLineEdit> #include <QPushButton>class Widget : public QWidget {Q_OBJECTpublic…