基础算法(4)——前缀和

news2024/11/17 11:41:23

1. 前缀和

题目描述:

解法一:暴力解法

直接模拟实现题目流程即可

时间复杂度为O(N*q),根据题目给出的条件q <= 10^{5},肯定会超时

解法二:前缀和(适用题型:快速 求出数组中某一个 连续区间 的 

时间复杂度:O(q)+O(n)

算法思路:

细节问题:为什么数组下标从 1 开始

当数组下标从 0 开始,dp[l - 1] 就成了 dp[-1] 造成数组越界异常

因此数组下标从 1 开始,若是数组下标从 0 开始时,需要处理边界情况

代码实现:

import java.util.Scanner;

// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Main {
    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        //1.读入数据
        int n = in.nextInt();
        int q = in.nextInt();
        int[] arr = new int[n + 1];
        for (int i = 1; i < n + 1; i++) {
            arr[i] = in.nextInt();
        }
        //2.预处理一个前缀和数组
        long[] dp = new long[n + 1]; //防溢出
        dp[1] = arr[1];
        for (int j = 2; j < n + 1; j++) {
            dp[j] += dp[j - 1] + arr[j];
        }
        //3.使用前缀和数组
        while (q > 0) {
            int l = in.nextInt();
            int r = in.nextInt();
            System.out.println(dp[r] - dp[l - 1]);
            q--;
        }
    }
}

2. 二维前缀和

题目描述:

解法一:暴力解法(模拟)

时间复杂度:O(n*m*q),会超时

解法二:前缀和

时间复杂度:O(m*n)+O(q)

算法思路:

代码实现:

import java.util.Scanner;

// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Main {
    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        //1.读入数据
        int n = in.nextInt();
        int m = in.nextInt();
        int q = in.nextInt();
        int[][] arr = new int[n + 1][m + 1];
        for (int i = 1; i < n + 1; i++) {
            for (int j = 1; j < m + 1; j++) {
                arr[i][j]= in.nextInt();
            }
        }
        //2.预处理一个前缀和矩阵
        long[][] dp = new long[n + 1][m + 1];
        for (int i = 1; i < n + 1; i++) {
            for (int j = 1; j < m + 1; j++) {
                dp[i][j] = dp[i - 1][j] + dp[i][j - 1] + arr[i][j] - dp[i - 1][j - 1];
            }
        }
        //3.使用前缀和矩阵
        while (q > 0) {
            int x1 = in.nextInt();
            int y1 = in.nextInt();
            int x2 = in.nextInt();
            int y2 = in.nextInt();
            System.out.println(dp[x2][y2] - dp[x1 - 1][y2] - dp[x2][y1 - 1] + dp[x1 - 1][y1 - 1]);
            q--;
        }
    }
}

3. 寻找数组的中心下标

题目描述:

解法:前缀和

算法思路:

从中心下标的定义可知,除中心下标的元素外,该元素左边的【前缀和】等于该元素右边的【后缀和】

因此我们可以先预处理出来两个数组,一个表示前缀和,一个表示后缀和

然后用一个 for 循环枚举可能的中心下标,判断每一个位置的【前缀和】以及【后缀和】,如果二者相等,就返回当前下标

代码实现:

class Solution {
    public int pivotIndex(int[] nums) {
        int n = nums.length;
        //f[i] 表示:[0, i - 1] 区间所有元素的和
        //g[i] 表示:[i + 1, n - 1] 区间所有元素的和
        int[] f = new int[n]; //前缀和数组
        int[] g = new int[n]; //后缀和数组

        //预处理前缀和后缀和数组(优化前)
        //这里当i等于0时,f[0]本来就是0,没必要多加判断,直接从1位置开始遍历即可
        // for (int i = 0; i < n; i++) {
        //     if (i == 0) {
        //         f[i] = 0;
        //     } else {
        //         f[i] = f[i - 1] + nums[i - 1];
        //     }
        // }
        //当i等于n-1时本来就是0,无需处理,直接从n-2处开始遍历即可
        // for (int i = n - 1; i >= 0; i--) {
        //     if (i == n - 1) {
        //         g[i] = 0;
        //     } else {
        //         g[i] = g[i + 1] + nums[i + 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++) {
            if (f[i] == g[i]) {
                return i;
            }
        }
        return -1;
    }
}

4. 除自身以外数组的乘积

题目描述:

解法:前缀积

算法思路:

根据题意,对于每一个位置的最终结果 answer[i] 都由两部分组成:

第一部分:nums[0] * nums[1] * nums[2] * ... * nums[i - 1]

第二部分:nums[i + 1] * nums[i + 2] * ... * nums[n - 1]

可以利用前缀和的思想,使用两个数组 f 和 g 分别处理出来两个信息:

代码实现:

class Solution {
    public int[] productExceptSelf(int[] nums) {
        int n = nums.length;
        //f[i] 表示[0, i-1] 区间内所有元素的乘积
        //g[i] 表示[i+1, n-1] 区间内所有元素的乘积
        int[] f = new int[n];
        int[] g = new int[n];
        int[] answer = new int[n];
        //细节问题,这两个下标处的值需设为1,设为0的话任何数乘0都得0了
        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++) {
            answer[i] = f[i] * g[i];
        }
        return answer;
    }
}

5. 和为 K 的子数组

题目描述:

解法一:暴力枚举

时间复杂度:O(N)

算法思路:

解法二:前缀和

算法思路:

代码实现:

class Solution {
    public int subarraySum(int[] nums, int k) {
        Map<Integer, Integer> hash = new HashMap<Integer, Integer>();
        hash.put(0, 1); //对应整个数组元素和为 k 的情况

        int sum = 0; //记录上一个位置的元素和
        int ret = 0; //记录前缀和为 sum - k 出现的次数
        for (int x : nums) {
            sum += x; //计算当前位置的前缀和
            ret += hash.getOrDefault(sum - k, 0); //统计结果
            hash.put(sum, hash.getOrDefault(sum, 0) + 1); //把当前的前缀和放入哈希表
        }
        return ret;
    }
}

6. 和可被 K 整除的子数组

题目描述:

知识补充:

算法思路:

代码实现:

class Solution {
    public int subarraysDivByK(int[] nums, int k) {
        Map<Integer, Integer> hash = new HashMap<Integer, Integer>();
        hash.put(0 % k, 1);

        int sum = 0; //标记前一个位置的前缀和
        int ret = 0; //表示最终结果
        for (int x : nums) {
            sum += x; //计算当前位置的前缀和
            int r = (sum % k + k) % k;
            ret += hash.getOrDefault(r, 0); //统计结果
            hash.put(r, hash.getOrDefault(r, 0) + 1);
        }
        return ret;
    }
}

7. 连续数组

题目描述:

解法:前缀和 + 哈希表

代码实现:

class Solution {
    public int findMaxLength(int[] nums) {
        Map<Integer, Integer> hash = new HashMap<>();
        hash.put(0, -1); //默认存在一个前缀和为 0 的情况

        int sum = 0;
        int ret = 0;
        for (int i = 0; i < nums.length; i++) { //此处要求下标,所以不能用 foreach
            //计算当前位置前缀和
            //无需修改原数组,仅需判断当前位置为 0 时,加上 -1 即可
            sum += (nums[i] == 0 ? -1 : 1);

            if (hash.containsKey(sum)) { //判断哈希表中是否存在前缀和
                ret = Math.max(ret, i - hash.get(sum));
            } else { //若哈希表中不存在前缀和,则更新
                hash.put(sum, i);
            }
        }
        return ret;
    }
}

实例:

8. 矩阵区域和

题目描述:

题目解析:

解法:二维前缀和

二维前缀和递推公式推导:

使用前缀和

算法思路:

此处将 answer 简写为 ans

第一步:当我们要求 ans[i][j] 时,仅需知道其对应的 x1、y1、x2、y2

第二步:下标的映射关系

代码实现:

class Solution {
    public int[][] matrixBlockSum(int[][] mat, int k) {
        int m = mat.length;
        int n = mat[0].length;

        //1.预处理前缀和矩阵
        int[][] dp = new int[m + 1][n + 1]; //此处 +1 操作就是为 dp 表增加一行一列
        for (int i = 1; i <= m; i++) { //dp 表从(1,1)开始
            for (int j = 1; j <= n; j++) {
                dp[i][j] = dp[i - 1][j] + dp[i][j - 1] - dp[i - 1][j - 1] + mat[i - 1][j - 1];
            }
        }

        //2.使用
        int[][] ret = new int[m][n];
        for (int i = 0; i < m; i++) {
            for (int j = 0; j < n; j++) {
                int x1 = Math.max(0, i - k) + 1;
                int y1 = Math.max(0, j - k) + 1;
                int x2 = Math.min(m - 1, i + k) + 1;
                int y2 = Math.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/2168254.html

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

相关文章

什么录屏软件最好?这四款软件留着有用!

在这个数字化时代&#xff0c;无论是教学分享、游戏直播还是产品演示&#xff0c;高质量的录屏软件都成为了我们不可或缺的工具。面对市面上琳琅满目的选择&#xff0c;到底哪款录屏软件才能真正满足你的需求&#xff0c;成为你创作路上的得力助手呢&#xff1f;别急&#xff0…

如何调用Ascend C算子

Ascend C是CANN针对算子开发场景推出的编程语言&#xff0c;原生支持C和C标准规范&#xff0c;兼具开发效率和运行性能。基于Ascend C编写的算子程序&#xff0c;通过编译器编译和运行时调度&#xff0c;运行在昇腾AI处理器上。使用Ascend C&#xff0c;开发者可以基于昇腾AI硬…

【ADC】ΔΣ ADC 中数字滤波器的延迟以及 SAR ADC 与 ΔΣ ADC 的差异对比总结

本文学习于TI 高精度实验室课程&#xff0c;深入探讨 delta-sigma 转换器中使用的数字滤波器。具体来说&#xff0c;本文将重点介绍数字滤波器如何引入延迟&#xff0c;因为这是 SAR 和 delta-sigma ADC 之间的显著差异。 文章目录 一、低延迟数字滤波器二、高延迟数字滤波器三…

MSVCR100.dll丢失怎么办,教你6种解决MSVCR100.dll丢失的方法

在计算机的日常使用中&#xff0c;我们可能会遇到各种各样的问题&#xff0c;其中之一就是MSVCR100.dll文件丢失。这个文件是Microsoft Visual C 2010的一个组件&#xff0c;如果丢失&#xff0c;可能会导致某些程序无法正常运行。那么&#xff0c;如何解决这个问题呢&#xff…

拉取指定版本的代码

// 获取指定版本的分支 https://git.swf.daimler.com/mbient/meta-mbient/-/blob/release/E066.1-2024.07.31-457-4870220-4882586/meta-mbient/recipes-mbient/dialog-domain-handlers/dialog-domain-handlers_git.bb?ref_typetags meta-mbient meta-mbient recipes-mbient …

大数据新视界 --大数据大厂之 Spark Streaming 实时数据处理框架:案例与实践

&#x1f496;&#x1f496;&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎你们来到 青云交的博客&#xff01;能与你们在此邂逅&#xff0c;我满心欢喜&#xff0c;深感无比荣幸。在这个瞬息万变的时代&#xff0c;我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…

如何通过10个简单步骤,让AI创作效率翻倍,性能提升90%

本文背景 随着不断深入地使用 AI 以及体验更多产品 最近对于大模型的使用感悟又有了一些新收获。 今天&#xff0c;特意来和大家分享 10 个大模型的使用妙招 。 这既是分享&#xff0c;也是我自己的学习梳理。 下面介绍的这些技巧&#xff0c;适用于所有大模型应用&#xff08;…

9月26日

1.虚函数与纯虚函数&#xff1a; 在类中定义函数时&#xff0c;在函数前加关键字 virtual &#xff0c;允许在派生类中重写的方法。那么该函数就是虚函数。 纯虚函数&#xff1a;没有实现的方法&#xff0c;用于定义接口。 2.基类为什么需要虚析构函数&#xff1a; 确保删除派生…

使用 Higress AI 插件对接通义千问大语言模型

前言 什么是 AI Gateway AI Gateway 的定义是 AI Native 的 API Gateway&#xff0c;是基于 API Gateway 的能⼒来满⾜ AI Native 的需求。例如&#xff1a; 将传统的 QPS 限流扩展到 token 限流。将传统的负载均衡/重试/fallback 能力延伸&#xff0c;支持对接多个大模型厂…

深入浅出 AbstractQueuedSynchronizer (AQS)

文章目录 什么是 AQSAQS 的工作原理同步状态&#xff08;state&#xff09;等待队列 AQS 是如何让线程排队并唤醒的公平锁和非公平锁AQS 的应用场景ReentrantLock&#xff08;可重入锁&#xff09;AQS 在 ReentrantLock 中的工作原理典型应用场景 CountDownLatch&#xff08;倒…

基于Django技术开发的酒店信息管理系统,包括员工用户功能和管理员用户功能两部分

项目摘要 该项目是基于Django技术开发的一套酒店管理系统&#xff0c;系统应用浏览器/服务期&#xff08;Browser/Server&#xff09;架构。系统主要包括员工用户功能和管理员用户功能两部分。开发员工信息管理、顾客信息管理、会员信息管理、停车场信息管理、餐厅信息管理、客…

HTML5+CSS3小实例:立方体控件的登录表单

实例:立方体控件的登录表单 技术栈:HTML+CSS 效果: 源码: 【HTML】 <!DOCTYPE html> <html lang="zh-CN"> <head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial…

【算法篇】二叉树类(1)(笔记)

目录 一、认识二叉树 1. 二叉树的种类 &#xff08;1&#xff09;满二叉树 &#xff08;2&#xff09;完全二叉树 &#xff08;3&#xff09;二叉搜索树 &#xff08;4&#xff09;平衡二叉搜索树 2. 二叉树的存储方式 3. 二叉树的遍历方式 4. 二叉树的定义 二、Leet…

(done) 使用泰勒展开证明欧拉公式

问问神奇的 GPT&#xff0c;how to prove euler formula? 一个答案如下&#xff1a;

华硕NUC亮相工博会,解锁工业AI PC解决方案

2024年9月24日至28日&#xff0c;中国国际工业博览会于上海国家会展中心盛大举行&#xff0c;华硕智能物联网展台位于展馆6.1H E183展位&#xff0c;在展位上华硕向大众展示了智能AI、物联网设备、华硕NUC等解决方案及IoT硬件产品&#xff0c;吸引了众多专业观众驻足交流和体验…

线程池的执行流程和配置参数总结

一、线程池的执行流程总结 提交线程任务&#xff1b;如果线程池中存在空闲线程&#xff0c;则分配一个空闲线程给任务&#xff0c;执行线程任务&#xff1b;线程池中不存在空闲线程&#xff0c;则线程池会判断当前线程数是否超过核心线程数&#xff08;corePoolSize&#xff09…

EfficientViT(2023CVPR):具有级联组注意力的内存高效视觉Transformer!

EfficientViT: Memory Efficient Vision Transformer with Cascaded Group Attention EfficientViT: 具有级联组注意力的内存高效视觉Transformer 万文长字&#xff0c;请耐心观看~ 论文地址&#xff1a; https://arxiv.org/abs/2305.07027 代码地址&#xff1a; Cream/Effici…

计算机毕业设计 饮食营养管理信息系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…

《强化学习的数学原理》(2024春)_西湖大学赵世钰 Ch9 策略梯度方法 -9.3.1

之前看了 2 次视频&#xff0c;公式有点多&#xff0c; 还是没整理出来。 这个版本是以下步骤后的版本 基本把相关的核心论文过了一遍&#xff0c;代码整理了部分 PDF 资料 整理 v3 链接 视频 链接 习题 策略梯度方法需要估计值 函数近似&#xff1a; 状态/动作 价值、策略 参…

C++初阶:STL详解(四)——vector迭代器失效问题

✨✨小新课堂开课了&#xff0c;欢迎欢迎~✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C&#xff1a;由浅入深篇 小新的主页&#xff1a;编程版小新-CSDN博客 一&#xff1a;迭代器失效的本质 迭代器的主…