LeetCode 每日一题笔记 日期:2026.05.17 题目:1306. 跳跃游戏 III

LeetCode 每日一题笔记

0. 前言

  • 日期:2026.05.17
  • 题目:1306. 跳跃游戏 III
  • 难度:中等
  • 标签:数组、深度优先搜索、广度优先搜索

1. 题目理解

问题描述

给定一个非负整数数组 arr,从下标 start 出发,每次可以从下标 i 跳到 i + arr[i]i - arr[i]

你无法跳出数组边界。判断是否能够跳到数组中任意一个值为 0 的下标。

示例

输入:arr = [4,2,3,0,3,1,2], start = 5

输出:true

解释:路径:5 → 4 → 1 → 3,下标3的值为0。

2. 解题思路

核心观察

  • 这是一个典型的图遍历问题,每个下标是节点,跳跃规则是边;
  • 需避免重复访问(否则会无限递归/循环);
  • 只要遍历过程中遇到值为0的节点,即可返回 true

算法步骤

  1. 检查当前下标是否值为0,若是直接返回 true
  2. 标记当前下标为已访问(防止循环);
  3. 尝试向右跳 i + arr[i],若合法且未访问,则递归;
  4. 尝试向左跳 i - arr[i],若合法且未访问,则递归;
  5. 若两条路径均失败,恢复标记并返回 false

3. 代码实现

java 复制代码
package lc1306;

public class Solution {
    public boolean canReach(int[] arr, int start) {
        if (arr[start] == 0) {
            return true;
        }
        int temp = arr[start];
        int n=start + arr[start];
        if (n< arr.length - 1 && arr[n] != -1) {
            arr[start] = -1;
            if (canReach(arr, n)) {
                return true;
            }
            arr[start]=temp;
        }
        int m=start - arr[start];
        if (m > 0 && m< arr.length - 1 && arr[m] != -1) {
            arr[start] = -1;
            if (canReach(arr, m)) {
                return true;
            }
            arr[start]=temp;
        }
        return false;
    }
}

4. 代码优化说明

减少了冗余的边界判断,统一处理左右跳转逻辑,同时去掉了不必要的 temp 变量与恢复操作,直接用访问标记避免循环:

java 复制代码
package lc1306;

import java.util.HashSet;
import java.util.Set;

public class Solution {
    public boolean canReach(int[] arr, int start) {
        return dfs(arr, start, new HashSet<>());
    }

    private boolean dfs(int[] arr, int index, Set<Integer> visited) {
        // 越界 或 已访问过,直接返回 false
        if (index < 0 || index >= arr.length || visited.contains(index)) {
            return false;
        }

        // 到达目标值 0,返回 true
        if (arr[index] == 0) {
            return true;
        }

        // 标记为已访问
        visited.add(index);

        int val = arr[index];
        // 左右两个方向只要有一条路通,就返回 true
        return dfs(arr, index + val, visited) || dfs(arr, index - val, visited);
    }
}

5. 复杂度分析

  • 时间复杂度 : O ( n ) O(n) O(n),每个下标最多访问一次;
  • 空间复杂度 : O ( n ) O(n) O(n),递归栈深度最坏为数组长度。

6. 总结

  • 核心思路是DFS 遍历 + 访问标记,利用数组原地修改标记访问状态;
  • 优化后代码逻辑更简洁,通过短路 || 直接返回结果,减少了分支判断;
  • 本题也可用 BFS 实现,思路一致,避免递归栈溢出风险。
相关推荐
Lsk_Smion21 小时前
力扣实训 _ [994].腐烂的橘子/图论
算法·leetcode·图论
自小吃多21 小时前
某志步进电机驱动器故障排查标准流程
笔记
zhangrelay21 小时前
后智能时代智能体推演预测娱乐文-节选-
笔记·学习·娱乐
8Qi81 天前
LeetCode 337:打家劫舍 III(House Robber III)—— 题解 ✅
算法·leetcode·二叉树·动态规划
2601_961194021 天前
教资科三美术考什么|初中高中美术题型考点和模板资料
leetcode·elasticsearch·职场和发展·蓝桥杯·pat考试·lucene
小碗羊肉1 天前
【Agent笔记 | 第六篇】Agent关键组件
笔记·agent
likerhood1 天前
服务器使用 vLLM 部署 Qwen2.5-Coder-7B-CL 笔记
服务器·笔记·vllm
Upsy-Daisy1 天前
Hermes Agent 学习笔记 01:一个会记忆、会学习、能长期运行的 AI Agent
人工智能·笔记·学习
8Qi81 天前
LeetCode 121 & 122:股票买卖问题(DP 对比题解)✅
算法·leetcode·职场和发展·动态规划
LuminousCPP1 天前
从零开始学 C++|系列开篇:从 C 到 C++ 的衔接之路
开发语言·c++·笔记