算法笔记(三)——前缀和算法

news2025/1/9 14:35:17

文章目录

  • 一维前缀和
  • 二维前缀和
  • 寻找数组的中心下标
  • 除自身以外数组的乘积
  • 和为 K 的子数组
  • 和可被 K 整除的子数组
  • 连续数组
  • 矩阵区域和

前缀和算法是一种用空间换时间的算法,他常常用于解决某些题目或者作为某些高级算法的组成部分

一维前缀和

题目链接:DP34 【模板】前缀和

在这里插入图片描述

思路

  • 通过数组arr存储输入的 n 个整数,数组 dp存储数组 arr的前缀和
  • 使用循环读取数组元素,并计算前缀和 dp
  • 进行q 次查询,每次查询给定一个区间[l, r]。查询结果为dp[r] - dp[l-1],表示数组在区间 [l, r] 的和

C++代码

#include<iostream>
using namespace std;

int main()
{
    int n, q;
    cin >> n >> q;
    long long arr[100001]={0}, dp[100001]={0};

    for(int i = 1; i <= n;++i)
    {
        cin >> arr[i];
        dp[i] = arr[i] + dp[i-1];
    }
    
    while(q--)
    {
        int l, r;
        cin >> l >> r;
        cout << dp[r] - dp[l-1] << endl;
    }
    
    return 0;
}

二维前缀和

题目:DP35 【模板】二维前缀和

在这里插入图片描述
思路
初始化前缀和

在这里插入图片描述

  • 构造前缀和数组dp[i][j],其含义是从(1,1)到(i,j)区域的面积;
  • D区域也是dp[i][j]存储的值,其面积也等于 A+B+C+D
  • 初始化前缀和dp[i][j] = (A+B)+(A+C)+D - A = dp[i-1][j]+dp[i][j-1]+arr[i][j]-dp[i-1][j-1]

使用前缀和计算

  • D = A+B+C+D-(A+B)-(A+C)+A
  • D = dp[i][j]-dp[i][j-1]-dp[i-1][j]+dp[i-1][j-1]

C++代码

#include <iostream>
using namespace std;

int arr[1100][1100];
long long dp[1100][1100];
int n, m, q, x1, x2, y1, y2;

int main() 
{
	cin >> n >> m >> q;
	for(int i = 1; i <=n; i++)
	{
		for(int j = 1; j <= m; j++)
		{
			cin >> arr[i][j];
            dp[i][j] = dp[i-1][j] + dp[i][j-1] + arr[i][j] - dp[i-1][j-1]; 
		}
	}

	while(q--)
	{
		cin >> x1 >> y1 >> x2 >> y2;
		cout << dp[x2][y2] - dp[x1-1][y2] - dp[x2][y1-1] + dp[x1-1][y1-1] << endl;
	}

	return 0;
}

寻找数组的中心下标

题目:寻找数组的中心下标

在这里插入图片描述
思路

  • 初始化f表示前缀和,g表示后缀和
  • f[i]表示[0,i-1]所有元素的和,
  • f[i] = f[i-1] + nums[i-1]
  • g[i]表示[i+1,n-1]所有元素的和,
  • g[i] = g[i+1] + nums[i+1];
  • 遍历数组nums,找到一个索引,使得 f[i](左侧元素的和)等于 l[i](右侧元素的和)

C++代码

class Solution {
public:
    int pivotIndex(vector<int>& nums) 
    {

        int n = nums.size();
        vector<int> f(n), g(n);

        for (int i = 1; i < n; i++)
            f[i] = f[i - 1] + nums[i - 1];
        for (int i = n - 2; i >= 0; i--)
            g[i] = g[i + 1] + nums[i + 1];

        for (int i = 0; i < n; i++)
            if (f[i] == g[i])
                return i;
           
        return -1;
    }
};

除自身以外数组的乘积

题目:除自身以外数组的乘积

在这里插入图片描述
思路

  • 初始化f表示前缀积,g表示后缀积
  • f[i]表示[0,i-1]所有元素的积,
  • f[i] = f[i - 1] * nums[i - 1]
  • g[i]表示[i+1,n-1]所有元素的积,
  • g[i] = g[i+1] * nums[i+1];
  • 遍历数组nums,计算res[i]的值

C++代码

class Solution 
{
public:
    vector<int> productExceptSelf(vector<int>& nums) 
    {
        int n = nums.size();

        vector<int> f(n + 1);
        vector<int> g(n + 1);
        vector<int> res(n);
        f[0] = 1,  g[n - 1] = 1; 

        for(int i = 1; i < n; i++)
            f[i] = f[i - 1] * nums[i - 1];

        for(int i = n - 2; i >= 0; i--)
            g[i] = g[i + 1] * nums[i + 1];

  
        for(int i = 0; i < n; i++)
            res[i] = f[i] * g[i];

        return res;  
    }
};

和为 K 的子数组

题目:和为 K 的子数组

在这里插入图片描述
思路
将问题转化为寻找和为k的子数组
而不是直接在数组中寻找和为k的连续元素,
这样就可以使问题在一次遍历中解决
具体来说就是:对于每个前缀和,都检查是否存在一个早先的前缀和,使得它们的差等于k。如果存在,就找到了一个和为k的子数组

  • 初始化一个哈希表 hash,其中hash[0]表示和为 0 的子数组的个数,初始化为 1
  • 两个变量 sumret,其中 sum 表示当前累积的和,ret 表示满足条件的子数组的个数
  • 遍历数组 nums,累积和并更新哈希表
    对于每个元素 x,更新 sum += x
    检查是否存在之前的累积和 sum - k 在哈希表中,如果存在,则累加 hash[sum - k] ret。更新哈希表中的当前累积和sum 的计数

C++代码

class Solution 
{
public:
    int subarraySum(vector<int>& nums, int k) 
    {
        // K:前缀和
        // V:前缀和出现的次数
        unordered_map<int, int> hash;
        int sum = 0, ans = 0;
        // 初始化时为空区间,则前缀和为0,出现的次数为1
        hash[0] = 1;
        for(int num : nums) 
        {
            sum += num; 
            ans += hash[sum - k]; // 要么为0,要么为sum - k出现的次数
            hash[sum]++;
        }
        return ans;
    }
};

和可被 K 整除的子数组

题目:和可被 K 整除的子数组

在这里插入**加粗样式**图片描述
补充

  1. 同余定理
    若(a-b) % p = k......0,则a % p = b % p
  2. C++、Java中 负数%正数等于负数
    修正 a % p + p
    为了a>0或a<0统一后(a % p + p) % p

思路
和上题基本相同,转化为在[0, i - 1]中,找到有多少个前缀和的余数等于(sum % k + k) % k

  • for(auto x : nums) 的循环中,遍历数组nums。对于每个元素x
  • sum += x; 累加当前位置的元素,得到当前位置的前缀和。
  • int r = (sum % k + k) % k
  • if(hash.count(r)) ret += hash[r]; 检查之前是否存在相同的余数r,如果存在,则将哈希表中对应的次数累加到结果 ret 中。
  • hash[r]++; 更新哈希表,将当前余数 r 的次数加一

C++代码

class Solution 
{
public:
    int subarraysDivByK(vector<int>& nums, int k) 
    {
        unordered_map<int, int> hash;
        hash[0] = 1; // 初始时,前缀和为 0 的余数的个数为 1 

        int sum = 0;
        int ret = 0;
        for(auto x : nums)
        {
            sum += x; // 当前位置前缀和
            int r = (sum % k + k) % k;
            if(hash.count(r)) // 检查之前是否存在相同的余数 r,如果存在
                ret += hash[r]; 

            hash[r]++; // 更新哈希表,将当前余数 r 的次数加一
        }

        return ret;
    }
};

连续数组

题目:连续数组

在这里插入图片描述
思路

  • 将 0 看作 -1,问题就转换成在数组中,找出最长的子数组使其和为0
  • unordered_map<int, int> hash表示创建一个哈希表,用于存储前缀和以及对应的出现位置
  • 遍历数组nums对于每个元素 nums[i],如果 nums[i] 是 0,则令 sum += -1;如果是1,则令 sum += 1。这样sum就表示当前位置的前缀和;
  • 判断哈希表中是否存在当前前缀和sum。如果存在,说明从上次该前缀和出现的位置到当前位置的子数组的和为零,更新最长长度 ret
  • 如果哈希表中不存在当前前缀和sum,则将当前前缀和和对应的位置存入哈希表中

C++代码

class Solution 
{
public:
    int findMaxLength(vector<int>& nums) 
    {
        unordered_map<int,int> hash;
        hash[0] = -1;

        int ret = 0;
        int sum = 0;
        for(int i = 0; i < nums.size(); ++i)
        {
            sum += (nums[i] == 1 ? 1 : -1);
            if(hash.count(sum)) 
                ret = max(ret, i - hash[sum]);
            else 
                hash[sum]=i;
        }

        return ret;
    }
};

矩阵区域和

题目:矩阵区域和

在这里插入图片描述
思路
二维前缀和问题,但是需要处理好边界问题

  • int m=mat.size(), n=mat[0].size()获取矩阵的行数和列数
  • vector<vector<int>> dp(m+1,vector<int>(n+1))创建二维前缀和数组dp
  • dp[i][j]表示矩阵左上角 (0,0)(i-1,j-1)的元素和,计算前缀和数组 dp
  • 对于每个位置(i,j),计算块的左上角和右下角的坐标,确保不超过矩阵边界

C++代码

class Solution 
{
public:
    vector<vector<int>> matrixBlockSum(vector<vector<int>>& mat, int k) 
    {
        int m = mat.size(), n = mat[0].size();
        vector<vector<int>> dp(m + 1, vector<int>(n + 1));
        for(int i = 1; i <=m; i++)
            for(int j = 1; j <= n; j++)
                    dp[i][j] = dp[i - 1][j] + dp[i][j - 1] + mat[i - 1][j - 1] - dp[i - 1][j - 1];

        vector<vector<int>> ret(m,vector<int>(n));
        for(int i = 0; i < m; i++)
            for(int j = 0; j < n; j++)
            {
                int x1 = max(0, i - k) + 1, y1 = max(0, j - k) + 1;
                int x2 = min(m - 1, i + k) + 1, y2 = min(n - 1, j + k) + 1;
                ret[i][j] = dp[x2][y2] - dp[x1 - 1][y2] - dp[x2][y1 - 1] + dp[x1 - 1][y1 - 1];
            }

        return ret;
    }
};

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

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

相关文章

【最新】微信小程序连接onenet——stm32+esp8266+onenet实现查看温湿度,控制单片机

微信小程序——stm32esp8266onenet实现查看温湿度&#xff0c;控制单片机 &#xff08;最新已验证&#xff09;stm32 新版 onenet dht11esp8266/01s mqtt物联网上报温湿度和控制单片机(保姆级教程) &#xff1a;↓↓&#x1f447; &#x1f447; &#x1f447; &#x1f447…

【Linux】进程优先级、调度、命令行参数:从理论到实践(二)

&#x1f308; 个人主页&#xff1a;Zfox_ &#x1f525; 系列专栏&#xff1a;Linux 目录 &#x1f680; 前言一&#xff1a; &#x1f525; 进程优先级 &#x1f375; 基本概念&#x1f375; 查看系统进程&#x1f375; PRI and NI&#x1f375; PRI vs NI&#x1f375; 用to…

【Java SE 题库】移除元素(暴力解法)--力扣

&#x1f525;博客主页&#x1f525;&#xff1a;【 坊钰_CSDN博客 】 欢迎各位点赞&#x1f44d;评论✍收藏⭐ 目录 1. 题目 2. 解法(快慢“指针”) 3. 源码 4. 小结 1. 题目 给你一个数组 nums 和一个值 val&#xff0c;你需要原地移除所有数值等于 val 的元素。元素的顺…

Pre-trained Models for Natural Language Processing: A Survey

前言 预训练模型给下游任务带来的效果不言而喻&#xff0c;有了预训练模型&#xff0c;我们可以使用它来加速解决问题的过程。正如论文中所说的那样&#xff0c;预训练模型&#xff08;PTMs&#xff09;的出现将自然语言处理&#xff08;NLP&#xff09;带入了一个新时代。本篇…

各种饺子的做法

【羊肉馅水饺】 材料:羊肉1000克、洋葱2个、香油3汤匙、盐适量、姜2片、料酒1汤匙、白胡椒粉、十三香1茶匙、 做法&#xff1a; 1.把羊肉剁成肉馅&#xff0c;羊肉选用带一些肥肉的&#xff0c;味道比较香&#xff0c;如果羊肉比较瘦&#xff0c;可以放一些猪的肥肉一起剁成馅…

【Python】探索自然语言处理的利器:THULAC 中文词法分析库详解

THULAC&#xff08;THU Lexical Analyzer for Chinese&#xff09;是清华大学开发的一款中文词法分析工具&#xff0c;集成了分词和词性标注两大功能。THULAC 拥有强大的分词能力和高效的词性标注&#xff0c;适用于多种中文文本处理场景。该工具能够在保证高准确率的同时保持较…

修复OpenSSH远程代码执行漏洞:版本升级到9.9p1

目录 前言1. 备份配置文件2. 下载 OpenSSH 最新版本3. 编译安装 OpenSSH4. 替换旧版 OpenSSH 并创建符号链接5. 重启 SSH 服务6. 验证安装结果结语参考文章 前言 OpenSSH 是一种广泛使用的远程登录协议&#xff0c;它确保了服务器和客户端之间的安全通信。然而&#xff0c;随着…

Linux驱动开发(速记版)--设备树

第五十二章 初识设备树 52.1 设备树介绍 设备树&#xff08;Device Tree&#xff09;是嵌入式系统和Linux内核中用于描述硬件的一种机制。 设备树概述 目的&#xff1a;描述硬件设备的特性、连接关系和配置信息。 优势&#xff1a;与平台无关&#xff0c;提高系统可移植性和可…

【linux进程】进程优先级命令行参数环境变量

目录 一&#xff0c;进程切换二&#xff0c;进程优先级1. 什么是优先级2. 权限 vs 优先级3. 为什么要有优先级4. 优先级的查看方式 三&#xff0c;命令行参数1. 什么是命令行参数2. 为什么要有命令行参数3. 是谁做的 四&#xff0c;环境变量1. 基本概念2. 常见环境变量3. 查看环…

电商店铺多开自动回复软件

在电商平台上开设多个店铺&#xff0c;即店铺多开&#xff0c;是一种扩展业务和增加销售额的策略。然而&#xff0c;店铺多开需要谨慎规划和执行&#xff0c;以避免违反平台规定和管理上的混乱。以下是如何实现店铺多开的详细步骤和注意事项。 1. 确定多开目标 在决定多开店铺…

[云] Getting Started with Kubernetes - Environment setup 环境配置

启动一个 EC2 实例&#xff0c; 请按照以下步骤操作&#xff1a; 登录 AWS 控制台&#xff1a; 打开浏览器&#xff0c;访问 AWS Management Console 并登录到您的账户。 选择 EC2 服务&#xff1a; 在 AWS 控制台的“服务”菜单中找到并点击“EC2”进入 EC2 Dashboard。 启…

Spark SQL分析层优化

导读&#xff1a;本期是《深入浅出Apache Spark》系列分享的第四期分享&#xff0c;第一期分享了Spark core的概念、原理和架构&#xff0c;第二期分享了Spark SQL的概念和原理&#xff0c;第三期则为Spark SQL解析层的原理和优化案例。本次分享内容主要是Spark SQL分析层的原理…

国内车牌号检测数据集 7800张 车牌识别 带标注 voc yolo

车牌号检测数据集 7800张 车牌识别 带标注 voc yolo 车牌号码检测数据集 名称 车牌号码检测数据集 (License Plate Detection Dataset) 规模 图像数量&#xff1a;7800张图像。类别&#xff1a;无分类&#xff0c;主要针对车牌区域的定位。标注个数&#xff1a;10,000多个标…

Windows 11 的 24H2 更新將帶來全新 Copilot+ AI PC 功能

10 月起率先向 Insider 推出。 微軟今日宣布 Windows 11 的 24H2 更新將從 10 月起陸續開放給不同用戶&#xff0c;其中將包括多項全新 Copilot AI PC 功能。首先是回爐重造的 Recall 回顧&#xff0c;官方在聽取社群意見後對安全性做了進一步加強。現在用戶可以自己選擇是否保…

从小IPD到大IPD

IPD体系实施或变革是一个长期的系统工程&#xff0c;需要坚持“总体规划、分步实施”的原则&#xff0c;业界经常有从“小IPD”到“大IPD”的说法。 像华为当年就是从“产品开发流程”开始推行&#xff0c;就是小范围的“小IPD”&#xff0c;后面逐步扩大为“大IPD”&#xff0…

Ubuntu22.04之mpv播放器高频快捷键(二百七十)

简介&#xff1a; CSDN博客专家、《Android系统多媒体进阶实战》一书作者 新书发布&#xff1a;《Android系统多媒体进阶实战》&#x1f680; 优质专栏&#xff1a; Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a; 多媒体系统工程师系列【…

两个向量所在平面的法线,外积,叉积,行列式

偶尔在一个数学题里面看到求两向量所在平面的法线&#xff0c;常规方法可以通过法线与两向量垂直这一特点&#xff0c;列两个方程求解&#xff1b;另外一种方法可以通过求解两个向量的叉积&#xff0c;用矩阵行列式 (determinant) 的方式&#xff0c;之前还没见过&#xff0c;在…

正则表达式调试工具实战

正则表达式调试工具实战 1、新建工程QWidget工程工程名RegexTool 如果QT不会配置,请参考我的博客,QT配置 Widget.cpp 默认内容如下 2、主界面设计 三行两列,每行采用HBoxLayout作为行布局控件,内部一个Lable控件和一个TextEdit控件,采用VBoxLayout 控件包裹三个HBoxLa…

JWT | JWT 漏洞介绍

关注这个漏洞的其他相关笔记&#xff1a;JWT 漏洞 - 学习手册-CSDN博客 0x01&#xff1a;JWT 漏洞介绍 JWT&#xff08;Json Web Token&#xff09;是用于在网络应用环境间传递声明的一种基于 JSON 的开放标准。它通过使用 JSON 对象进行安全的信息传输&#xff0c;可以用于身…

ChatGPT实时语音将于本周向免费用户推出:OpenAI DevDay 2024详细解读

大家好&#xff0c;我是木易&#xff0c;一个持续关注AI领域的互联网技术产品经理&#xff0c;国内Top2本科&#xff0c;美国Top10 CS研究生&#xff0c;MBA。我坚信AI是普通人变强的“外挂”&#xff0c;专注于分享AI全维度知识&#xff0c;包括但不限于AI科普&#xff0c;AI工…