力扣每日一题 2765最长交替子数组

本文探讨如何在给定整数数组中找到最长的交替子数组,提供Java和Python代码实现,并优化了算法效率。
摘要由CSDN通过智能技术生成

2765. 最长交替子数组

题目描述:

给你一个下标从 0 开始的整数数组 nums 。如果 nums 中长度为 m 的子数组 s 满足以下条件,我们称它是一个 交替子数组 :

  • m 大于 1 。
  • s1 = s0 + 1 。
  • 下标从 0 开始的子数组 s 与数组 [s0, s1, s0, s1,...,s(m-1) % 2] 一样。也就是说,s1 - s0 = 1 ,s2 - s1 = -1 ,s3 - s2 = 1 ,s4 - s3 = -1 ,以此类推,直到 s[m - 1] - s[m - 2] = (-1)^m 。

请你返回 nums 中所有 交替 子数组中,最长的长度,如果不存在交替子数组,请你返回 -1 。

子数组是一个数组中一段连续 非空 的元素序列。

示例 1:

输入:nums = [2,3,4,3,4]
输出:4
解释:交替子数组有 [3,4] ,[3,4,3] 和 [3,4,3,4] 。最长的子数组为 [3,4,3,4] ,长度为4 。

示例 2:

输入:nums = [4,5,6]
输出:2
解释:[4,5] 和 [5,6] 是仅有的两个交替子数组。它们长度都为 2 。

提示:

  • 2 <= nums.length <= 100
  • 1 <= nums[i] <= 10^4

思路:

双重循环,外层循环寻找子数组的头,内层循环寻找最大的符合要求的点在哪里。
关键在于对于跳出循环以后的重置操作放在哪,以及如何重置,非常的重要,可以用不断debug的方式去完成。

代码:

java里得到数组长度:nums.length

class Solution {
    public int alternatingSubarray(int[] nums) {
    int max_len=1;//目前最长的长度
    int len=1;//当前长度
    int flag=1;//下一个元素应该与当前元素差flag
    for(int i=0;i<nums.length;i++){
        //外层循环,相当于在寻找子数组的头
        len=1;//每一轮结束,要将当前长度与标志重置
        flag=1;
        for(int j =i+1;j<nums.length;j++){
            System.out.println(flag);
            if(nums[j]-nums[j-1]==flag){
                flag=(-1)*flag;
                len+=1;
                if(len>max_len && len<=nums.length)
                {max_len=len;
                System.out.println(max_len);
                }
            }
            else{
                len=1;//重置
                flag=1;
                break;
            }
        }
    }
    if(max_len==1){
        return -1;
    }
    else{
        return max_len;
    }
    }
}

官解:

方法一比较直观,但是复杂度比较高。当内层循环判断出当前下标 i打破了交替子数组的条件后,我们其实不需要回到外层循环的下标 firstIndex,因为通过内层循环,我们已经知道子数组 nums[firstIndex,…,i−1]是满足交替子数组的条件的。分析以下情况:

如果这个子数组的长度大于等于 3,那么我们不需要从 firstIndex+1开始外层循环,因为子数组 nums[firstIndex+1,firstIndex+2]必不满足交替子数组。
如果这个子数组的长度大于等于 4,那么我们不需要从 firstIndex+2 开始外层循环,因为子数组 nums[firstIndex+2,…,i−1]虽然满足交替子数组,但是这个交替数组会在 i被破环,并且长度小于 nums[firstIndex,…,i−1]。
通过这样分析,我们可以得出结论,外层的循环可以从 i−1继续。而这样的话,我们可以丢弃外层循环,在内层循环多做一个 firstIndex是否可以为 i−1的判断,而只保留内层循环。如果可以,则将 firstIndex 更新为 i−1,并更新最长长度。如果不可以,则将 firstIndex更新为 i。两种情况下,内层循环下一个遍历的下标都为 i+1,这样,我们就只用循环一层,即可以在循环结束后返回结果。

python代码:

class Solution:
    def alternatingSubarray(self, nums: List[int]) -> int:
        res = -1
        firstIndex = 0
        for i in range(1, len(nums)):
            length = i - firstIndex + 1
            if nums[i] - nums[firstIndex] == (length - 1) % 2:
                res = max(res, length)
            else:
                if nums[i] - nums[i - 1] == 1:
                    firstIndex = i - 1
                    res = max(res, 2)
                else:
                    firstIndex = i
        return res

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值