代码随想录刷题笔记-Day33

1. 跳跃游戏

55. 跳跃游戏icon-default.png?t=N7T8https://leetcode.cn/problems/jump-game/

给你一个非负整数数组 nums ,你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。

判断你是否能够到达最后一个下标,如果可以,返回 true ;否则,返回 false 。

示例 1:

输入:nums = [2,3,1,1,4]
输出:true
解释:可以先跳 1 步,从下标 0 到达下标 1, 然后再从下标 1 跳 3 步到达最后一个下标。

示例 2:

输入:nums = [3,2,1,0,4]
输出:false
解释:无论怎样,总会到达下标为 3 的位置。但该下标的最大跳跃长度是 0 , 所以永远不可能到达最后一个下标。

解题思路

要得到的是一个最大可到达范围,局部的最优是当前的到达最大范围

代码

class Solution {
	public boolean canJump(int[] nums) {
		if (nums.length == 1)
			return true;
		int cover = 0;
		for (int i = 0; i <= cover; i++) {
			cover = Math.max(cover, i+nums[i]);
			if (cover >= nums.length - 1)
				return true;
		}

		return false;
	}
}

2. K次取反后最大化的数组和

1005. K 次取反后最大化的数组和icon-default.png?t=N7T8https://leetcode.cn/problems/maximize-sum-of-array-after-k-negations/

给你一个整数数组 nums 和一个整数 k ,按以下方法修改该数组:

  • 选择某个下标 i 并将 nums[i] 替换为 -nums[i] 。

重复这个过程恰好 k 次。可以多次选择同一个下标 i 。

以这种方式修改数组后,返回数组 可能的最大和 。

示例 1:

输入:nums = [4,2,3], k = 1
输出:5
解释:选择下标 1 ,nums 变为 [4,-2,3] 。

示例 2:

输入:nums = [3,-1,0,2], k = 3
输出:6
解释:选择下标 (1, 2, 2) ,nums 变为 [3,1,0,2] 。

示例 3:

输入:nums = [2,-3,-1,5,-4], k = 2
输出:13
解释:选择下标 (1, 4) ,nums 变为 [2,3,-1,5,4] 。

解题思路

必须操作K次,所以,先把所有的负数给转为正数,然后,如果有剩余次数,在最小数的位置判断剩余次数的奇偶。

代码

class Solution {
    public int largestSumAfterKNegations(int[] nums, int k) {
        Arrays.sort(nums);
        int sum = 0;
        int index = 0;
        int min = nums[0] < 0 ? -nums[0] : nums[0];
        while (index < nums.length) {
            if (k > 0) {
                if (nums[index] >= 0) {// k未用完遇到了正数
                    if (min <= nums[index]) {// 如果最小值已经错过了,就判断是不做操作还是减去两倍的最小值
                        sum -= k % 2 == 0 ? 0 : 2 * min;
                        sum += nums[index];
                    } else {// 如果是当前,就判断是加还是减
                        sum += k % 2 == 0 ? nums[index] : -nums[index];
                    }
                    k = 0;
                } else {// k未用完并且是负数,那就加上
                    sum += min = -nums[index];
                    k--;
                }
            } else {// k用完了,正常累加
                sum += nums[index];
            }
            index++;
        }
        if (k > 0)// 未遇到正数,所以k没有经历一次用完,需要进行处理
            sum -= k % 2 == 0 ? 0 : 2 * min;
        return sum;
    }
}

3. 跳跃游戏 II

45. 跳跃游戏 IIicon-default.png?t=N7T8https://leetcode.cn/problems/jump-game-ii/

给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]

每个元素 nums[i] 表示从索引 i 向前跳转的最大长度。换句话说,如果你在 nums[i] 处,你可以跳转到任意 nums[i + j] 处:

  • 0 <= j <= nums[i] 
  • i + j < n

返回到达 nums[n - 1] 的最小跳跃次数。生成的测试用例可以到达 nums[n - 1]

示例 1:

输入: nums = [2,3,1,1,4]
输出: 2
解释: 跳到最后一个位置的最小跳跃数是 2。
     从下标为 0 跳到下标为 1 的位置,跳 1 步,然后跳 3 步到达数组的最后一个位置。

示例 2:

输入: nums = [2,3,0,1,4]
输出: 2

解题思路

对于每一步,只需要记录到达这里的最少步数。第一想法是DP,使用一个数组进行记录。但是好像不用,在第一步范围内的为1步,在第1步骤内延伸出去的都为2步,在2步延伸出去的部分所延伸出去的都为第三步。

代码

class Solution {
	public int jump(int[] nums) {
		int step = 0;
		int cur = 0;
		int next = 0;
		for (int i = 0; i < nums.length; i++) {
            if (cur >= nums.length - 1)
				return step;
			next = Math.max(next, nums[i] + i);
			if (i == cur) {//如果step步可以到达的区域遍历完了,那么该进入step+1步可以到达的区域了,需要更新cur和next做边界
				cur = next;
				step++;
			}
		}

		return step;
	}
}

相关推荐

  1. 代码随想笔记Day36

    2024-03-12 04:28:03       29 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-03-12 04:28:03       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-03-12 04:28:03       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-03-12 04:28:03       19 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-03-12 04:28:03       20 阅读

热门阅读

  1. ARM TrustZone技术介绍

    2024-03-12 04:28:03       18 阅读
  2. linux新一代的RPM软件包管理器dnf

    2024-03-12 04:28:03       27 阅读
  3. Linux中basename作用

    2024-03-12 04:28:03       24 阅读
  4. Dutree:Linux 文件系统磁盘使用追踪工具

    2024-03-12 04:28:03       20 阅读
  5. 权限管理系统-0.3.0

    2024-03-12 04:28:03       20 阅读
  6. 【Flink SQL】Flink SQL 基础概念:数据类型

    2024-03-12 04:28:03       21 阅读
  7. Vuex getters源码分析

    2024-03-12 04:28:03       19 阅读