【前后缀】Leetcode hot 100

文章目录

  • [LeetCode 238.除自身以外数组的乘积](#LeetCode 238.除自身以外数组的乘积)
  • [LeetCode 560.和为K的子数组](#LeetCode 560.和为K的子数组)
  • [LeetCode 121.买卖股票的最佳时机](#LeetCode 121.买卖股票的最佳时机)
  • [LeetCode 53.最大子数组和](#LeetCode 53.最大子数组和)

接雨水也算前后缀的题,只是在双指针那和盛水最多的容器写在了一起。

LeetCode 238.除自身以外数组的乘积

java 复制代码
class Solution {
    public int[] productExceptSelf(int[] nums) {
        int n = nums.length;
        int[] pre = new int[n];
        pre[0] = 1;
        for(int i = 1; i < n; i++){
            pre[i] = pre[i-1] * nums[i-1];
        }

        int[] suf = new int[n];
        suf[n-1] = 1;
        for(int i = n-2; i >= 0; i--){//注意这里是大于等于0,i--
            suf[i] = suf[i+1] * nums[i+1];
        }

        int[] ans = new int[n];
        for(int i = 0; i < n; i++){
            ans[i] = pre[i] * suf[i];
        }
        
        return ans;
    }
}

LeetCode 560.和为K的子数组

java 复制代码
class Solution {
    public int subarraySum(int[] nums, int k) {
        //sum(i,j)=k,也就是
        // s[j]-s[i-1] = k
        // s[i-1] = s[j]-k
        // 在 j 之前,有多少个 i-1 存在,使得 s[i-1] 的值恰好等于 s[j] - k。这里的s[j]包含nums[j]这个值!!和之前除自身以外数组的乘积那个题不一样!那个是左边所有元素的乘积
        int ans = 0;
        Map<Integer,Integer> prefixSumCounts = new HashMap<>();

        //注意这步
        prefixSumCounts.put(0, 1); //为了让s[i-1] = s[0-1] = s[-1]这种情况合法,因为要找0-j-1这些i里面符合s[j]-s[i-1] = k的数

        //用来维护s[j]的动态变化
        int currentPrefixSum = 0;

        for(int x:nums){
            //算s[j]
            currentPrefixSum += x;

            //算s[i-1]
            int targetPrefixSum = currentPrefixSum - k;
            //如果上面这个东西是0,同时x还是nums的第一个值,是存在那么1个的

            //找这个符合s[i-1]=s[j]-k(k=s[j]-s[i-1])的s[i-1]是否存在
            if(prefixSumCounts.containsKey(targetPrefixSum)){
                int count = prefixSumCounts.get(targetPrefixSum);
                ans+=count;
            }

            //存储s[j]
            // (1) 先获取 "currentPrefixSum" 之前出现过几次(默认为 0)
            int currentPrefixSumCnt = 0;
            if(prefixSumCounts.containsKey(currentPrefixSum)){
                currentPrefixSumCnt = prefixSumCounts.get(currentPrefixSum);
            }

            // (2) 在旧次数上 +1,然后存回哈希表
            prefixSumCounts.put(currentPrefixSum, currentPrefixSumCnt + 1);
        }
        return ans;
    }
}

思路总结:

其实这个没有维护一个前缀和数组,而是就是利用一个动态的值一次遍历计算前缀和,创造出了一个pre[nums[-1]]=0以便于求差计算(尤其是在子数组的情况下)!!因为会出现pre[n]-pre[0-1]也就是pre[n]-pre[-1]的情况

LeetCode 121.买卖股票的最佳时机

这题其实不算前缀和,算贪心,但是因为和53比较相似,也放进来了

java 复制代码
class Solution {
    public int maxProfit(int[] prices) {
        int ans = 0;
        int minPrice = prices[0];

        for(int p:prices){
            //注意先更新ans,后更新minPrice
            //先看如果今天卖能赚多少
            //再看要不要把今天当作新的最低价买点,使得后续能够用今天的价格计算
            //不然如果今天是历史最低点,这样算只考虑了今天买今天卖的情况,没有考虑之前买今天卖的情况
            ans = Math.max(ans, p-minPrice);
            minPrice = Math.min(minPrice,p);
        }
        return ans;
    }
}

思路总结:

主要就是注意先更新ans,后更新minPrice。

如果今天是历史最低点,先更新minPrice的话,只考虑了今天买今天卖的情况,没有考虑之前买今天卖的情况。不过题目也有要求,"你只能选择 某一天 买入这只股票,并选择在 未来的某一个不同的日子 卖出该股票。"

LeetCode 53.最大子数组和

java 复制代码
class Solution {
    public int maxSubArray(int[] nums) {
        int ans = Integer.MIN_VALUE;
        int minPreSum = 0;
        int preSum = 0;

        for(int x : nums){
            preSum += x;
            //先更新ans,后更新minPreSum
            //不然类比股票,只考虑了今天买今天卖的情况,没有考虑之前买今天卖的情况
            //而且这样会把当前前缀和误当成最小前缀和,导致计算出非法的空子数组,因为题目约束子数组最小为1
            ans = Math.max(ans, preSum-minPreSum); // 减去前缀和的最小值
            minPreSum = Math.min(minPreSum,preSum);// 维护前缀和的最小值
        }

        return ans;
    }
}

思路总结:

一边遍历数组计算前缀和,一边维护前缀和的最小值,用当前的前缀和减去前缀和的最小值,就得到了以当前元素结尾的子数组和的最大值,用它来更新答案的最大值。

由于题目要求子数组不能为空,应当先计算前缀和-最小前缀和,再更新最小前缀和。相当于不能在同一天买入股票又卖出股票。

相关推荐
wand codemonkey12 分钟前
SpringbootWeb【入门】+MySQL【安装】+【DataDrip安装 】+【连接MySQL】
java·mysql·mybatis
Mahir088 小时前
Spring 循环依赖深度解密:从问题本质到三级缓存源码级解析
java·后端·spring·缓存·面试·循环依赖·三级缓存
RyFit9 小时前
SpringAI 常见问题及解决方案大全
java·ai
石山代码9 小时前
C++ 内存分区 堆区
java·开发语言·c++
心中有国也有家10 小时前
cann-recipes-infer:昇腾 NPU 推理的“菜谱集合”
经验分享·笔记·学习·算法
绝知此事10 小时前
【算法突围 01】线性结构与哈希表:后端开发的收纳术
java·数据结构·算法·面试·jdk·散列表
无风听海10 小时前
C# 隐式转换深度解析
java·开发语言·c#
碧海银沙音频科技研究院10 小时前
通话AEC与语音识别AEC的软硬回采链路
深度学习·算法·语音识别
一只大袋鼠10 小时前
Git 进阶(二):分支管理、暂存栈、远程仓库与多人协作
java·开发语言·git
csdn_aspnet11 小时前
Python 算法快闪 LeetCode 编号 70 - 爬楼梯
python·算法·leetcode·职场和发展