找实习之从0开始的后端学习日记【9.5】

918. 环形子数组的最大和

给定一个长度为 n 的环形整数数组 nums ,返回 nums 的非空 子数组 的最大可能和 。

环形数组 意味着数组的末端将会与开头相连呈环状。形式上, nums[i] 的下一个元素是 nums[(i + 1) % n] , nums[i] 的前一个元素是 nums[(i - 1 + n) % n] 。

子数组 最多只能包含固定缓冲区 nums 中的每个元素一次。形式上,对于子数组 nums[i], nums[i + 1], …, nums[j] ,不存在 i <= k1, k2 <= j 其中 k1 % n == k2 % n 。

示例 1:

输入:nums = [1,-2,3,-2]
输出:3
解释:从子数组 [3] 得到最大和 3
示例 2:

输入:nums = [5,-3,5]
输出:10
解释:从子数组 [5,5] 得到最大和 5 + 5 = 10
示例 3:

输入:nums = [3,-2,2,-3]
输出:3
解释:从子数组 [3] 和 [3,-2,2] 都可以得到最大和 3

提示:

n == nums.length
1 <= n <= 3 * 104
-3 * 104 <= nums[i] <= 3 * 104​​​​​​​

来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/maximum-sum-circular-subarray
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

我的解法:分类讨论+动规
在这里插入图片描述
其中f和b都使用了滚动数组来省空间。

class Solution {
    public int maxSubarraySumCircular(int[] nums) {
    int n = nums.length;
    int maxResult = nums[0];
    int pre = 0;//f0
    int cur = nums[0];//f0
    int bpre = 0;
    int bcur = 0;
    int[] a = new int[n];
    // int[] b = new int[n];
    int[] g = new int[n];
    for(int i = 0;i<n;i++){
        // 1 - fi
        cur = Math.max(nums[i],pre+nums[i]);
        maxResult = Math.max(cur,maxResult);
        
        pre = cur;

        // 2 - ai bi gi
        if(i==0){
            a[i] = nums[i];
            // b[n-1-i] = nums[n-1-i];
            bcur = nums[n-1-i];
            g[n-1-i] = nums[n-1-i];

        }
        else{
            a[i] = nums[i]+a[i-1];
            // b[n-1-i] = nums[n-1-i]+b[n-i];
            bcur = bpre +  nums[n-1-i];
            g[n-1-i] = Math.max(bcur,g[n-i]);
        }
        bpre = bcur;
    }
    for(int i=0;i<n;i++){
        if(i==n-1){
            maxResult = Math.max(maxResult,a[i]);
        }  
        else{
        maxResult = Math.max(maxResult,a[i]+g[i+1]);
        }}
        
    return maxResult;
    }
}

题解 前缀和 + 优先队列

一切建立在把环形→A+A的思想上。可以学习这种思维。
还可以记录一下java中队列的写法(用Deque接收ArrayDeque)。
在这里插入图片描述


class Solution {
    public int maxSubarraySumCircular(int[] A) {
        int N = A.length;

        // Compute P[j] = B[0] + B[1] + ... + B[j-1]
        // for fixed array B = A+A
        int[] P = new int[2*N+1];
        for (int i = 0; i < 2*N; ++i)
            P[i+1] = P[i] + A[i % N];

        // Want largest P[j] - P[i] with 1 <= j-i <= N
        // For each j, want smallest P[i] with i >= j-N
        int ans = A[0];
        // deque: i's, increasing by P[i]
        Deque<Integer> deque = new ArrayDeque();
        deque.offer(0);

        for (int j = 1; j <= 2*N; ++j) {
            // If the smallest i is too small, remove it.
            if (deque.peekFirst() < j-N)
                deque.pollFirst();

            // The optimal i is deque[0], for cand. answer P[j] - P[i].
            ans = Math.max(ans, P[j] - P[deque.peekFirst()]);

            // Remove any i1's with P[i2] <= P[i1].
            while (!deque.isEmpty() && P[j] <= P[deque.peekLast()])
                deque.pollLast();

            deque.offerLast(j);
        }

        return ans;
    }
}

作者:LeetCode
链接:https://leetcode.cn/problems/maximum-sum-circular-subarray/solution/huan-xing-zi-shu-zu-de-zui-da-he-by-leetcode/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。```

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值