403. 青蛙过河 (Hard)
问题描述
一只青蛙想要过河。 假定河流被等分为若干个单元格,并且在每一个单元格内都有可能放有一块石子(也有可能没有)。
青蛙可以跳上石子,但是不可以跳入水中。
给你石子的位置列表 stones(用单元格序号 升序 表示),
请判定青蛙能否成功过河(即能否在最后一步跳至最后一块石子上)。开始时,
青蛙默认已站在第一块石子上,并可以假定它第一步只能跳跃 1 个单位(即只能从单元格 1 跳至单元格 2 )。
如果青蛙上一步跳跃了 k 个单位,那么它接下来的跳跃距离只能选择为 k - 1、 k 或 k + 1
个单位。 另请注意,青蛙只能向前方(终点的方向)跳跃。
示例 1:
输入:stones = [0,1,3,5,6,8,12,17]
输出:true
解释:青蛙可以成功过河,按照如下方案跳跃:跳 1 个单位到第 2 块石子, 然后跳 2 个单位到第 3 块石子, 接着
跳 2 个单位到第 4 块石子, 然后跳 3 个单位到第 6 块石子, 跳 4 个单位到第 7 块石子, 最后,跳 5
个单位到第 8 个石子(即最后一块石子)。
示例 2:
输入:stones = [0,1,2,3,4,8,9,11]
输出:false
解释:这是因为第 5 和第 6 个石子之间的间距太大,没有可选的方案供青蛙跳跃过去。
提示:
2 <= stones.length <= 20000 <= stones[i] <= 2³¹ - 1stones[0] == 0stones按严格升序排列
解题思路
记忆化搜索
我们考虑dfs(i, k)表示从第i个石子跳k步,之后能否到达终点;
- 如果从第
i个石子跳k步,不能到达另一个石子,return false;; - 否则,记从第
i个石子跳k步到达的新石头的索引为new_idx,那么只要从new_idx跳k + 1,k,k - 1任意一步能到达终点,则dfs(i, k)返回的结果为true。 
边界条件if (idx == stones.size() - 1) return true;,同时k不能为0, 为0则return false;
动态规划
记dp[i][k]为到达从上一个石子处跳k个单位到达第i个石子(注意这里的上一个石子并不一定是第i - 1石子,而是stones[i] - k位置对应的的石子,记该索引为pre_idx = ump[stones[i] - k]),对应的状态转移方程为:
dp[i] = dp[pre_idx][k] || dp[pre_idx][k - 1] || dp[pre_idx][k + 1];,这里dp[i][k]应该初始化为false, 同时dp[0][0] = true;
bfs
其实就是记忆化搜索的翻版,visited数组变成vector<vector<bool>> visited(stones.size(), vector<bool>(stones.size() + 1, false));,分别表示石子坐标和到达该坐标的步数;
注意pair入队时,要更新visited数组
bfs
代码
记忆化搜索
class Solution {
  public:
    bool dfs(int start_idx, int mv_step, vector<int> &stones, unordered_map<int, int> &ump, vector<vector<int>> &cache) {
        if (start_idx == stones.size() - 1) {
            return true; // ?这里不确定
        }
        if (mv_step <= 0) {
            return false;
        }
        if (ump.find(stones[start_idx] + mv_step) != ump.end()) {
            if (cache[start_idx][mv_step] > -1)
                return cache[start_idx][mv_step];
            int new_idx = ump[stones[start_idx] + mv_step];
            cache[start_idx][mv_step] = dfs(new_idx, mv_step - 1, stones, ump, cache) || dfs(new_idx, mv_step, stones, ump, cache) || dfs(new_idx, mv_step + 1, stones, ump, cache);
            return cache[start_idx][mv_step];
        }
        return false;
    }
    bool canCross(vector<int> &stones) {
        // 尝试记忆化搜索的写法
        unordered_map<int, int> ump;
        for (int i = 0; i < stones.size(); ++i) {
            ump[stones[i]] = i;
        }
        vector<vector<int>> cache(stones.size(), vector<int>(stones.size() + 1, -1));
        return dfs(0, 1, stones, ump, cache);
    }
};
动态规划
class Solution {
  public:
    bool canCross(vector<int> &stones) {
        unordered_map<int, int> ump;
        for (int i = 0; i < stones.size(); ++i) {
            ump[stones[i]] = i;
        }
        // 跳了k步,到达stones[i], dp[i][k];
        vector<vector<bool>> dp(stones.size(), vector<bool>(stones.size() + 1, false));
        dp[0][0] = true;
        for (int i = 1; i < stones.size(); ++i) {
            for (int k = 1; k <= i; ++k) {
                if (ump.find(stones[i] - k) != ump.end()) {
                    int pre_idx = ump[stones[i] - k];
                    dp[i][k] = dp[pre_idx][k] || dp[pre_idx][k - 1] || dp[pre_idx][k + 1];
                }
            }
        }
        for (int k = 1; k < stones.size(); ++k) {
            if (dp[stones.size() - 1][k]) {
                return true;
            }
        }
        return false;
    }
};
bfs
class Solution {
  public:
    bool canCross(vector<int> &stones) {
        if (stones[1] > 1) {
            return false;
        }
        vector<vector<bool>> visited(stones.size(), vector<bool>(stones.size() + 1, false));
        visited[1][1] = true;
        unordered_map<int, int> ump;
        for (int i = 0; i < stones.size(); ++i) {
            ump[stones[i]] = i;
        }
        queue<pair<int, int>> q;
        q.push({1, 1});
        while (!q.empty()) {
            auto [idx, mv_step] = q.front();
            q.pop();
            if (idx == stones.size() - 1)
                return true;
            for (int i = mv_step + 1; i > 0 && i >= mv_step - 1; --i) {
                if (ump.find(stones[idx] + i) != ump.end()) {
                    int new_idx = ump[stones[idx] + i];
                    if (visited[new_idx][i] == false) { // 说明这个点没有被访问过
                        visited[new_idx][i] = true;
                        q.push({new_idx, i});
                    }
                }
            }
        }
        return false;
    }
};
												
											403. 青蛙过河 (Hard)的更多相关文章
- Java实现 LeetCode 403 青蛙过河
		
403. 青蛙过河 一只青蛙想要过河. 假定河流被等分为 x 个单元格,并且在每一个单元格内都有可能放有一石子(也有可能没有). 青蛙可以跳上石头,但是不可以跳入水中. 给定石子的位置列表(用单元格序 ...
 - [LeetCode] Frog Jump 青蛙过河
		
A frog is crossing a river. The river is divided into x units and at each unit there may or may not ...
 - NOIP 2005 青蛙过河
		
做题记录:2016-08-10 21:58:09 题目描述 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都 ...
 - ooj 1066 青蛙过河DP
		
http://121.249.217.157/JudgeOnline/problem.php?id=1066 1066: 青蛙过河 时间限制: 1 Sec 内存限制: 64 MB提交: 58 解决 ...
 - 趣味算法——青蛙过河(JAVA)
		
青蛙过河是一个非常有趣的智力游戏,其大意如下: 一条河之间有若干个石块间隔,有两队青蛙在过河,每队有3只青蛙,这些青蛙只能向前移动,不能向后移动,且一次只能有一只青蛙向前移动.在移动过程中,青蛙可以向 ...
 - HRBUST 1186 青蛙过河 (思路错了)
		
在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙可能到达的点看成数轴上的一串 ...
 - P1244 青蛙过河
		
P1244 青蛙过河NOI2000主要思想:数学归纳法 递推 压位高精度 化归 理解能力和找规律的能力题意再述:1.青蛙从上到下必须连续递增或者下面是石墩 而不能是1 12 33 4而且每时每刻都要满 ...
 - 洛谷P1244 青蛙过河
		
P1244 青蛙过河 362通过 525提交 题目提供者该用户不存在 标签 难度普及- 时空限制1s / 128MB 提交 讨论 题解 最新讨论更多讨论 题目什么意思 题目看不懂啊 题目描述 有一条河 ...
 - 洛谷 P1244 青蛙过河
		
P1244 青蛙过河 题目描述 有一条河,左边一个石墩(A区)上有编号为1,2,3,4,…,n的n只青蛙,河中有k个荷叶(C区),还有h个石墩(D区),右边有一个石墩(B区),如下图所示.n只青蛙要过 ...
 - 洛谷P1244 [NOI2000] 青蛙过河  [2017年4月计划 动态规划07]
		
P1244 青蛙过河 题目描述 有一条河,左边一个石墩(A区)上有编号为1,2,3,4,…,n的n只青蛙,河中有k个荷叶(C区),还有h个石墩(D区),右边有一个石墩(B区),如下图所示.n只青蛙要过 ...
 
随机推荐
- Go语言与其他高级语言的区别
			
概述: go语言与其他语言相比,go语言的关键字非常少,只有25个,c语言有37个,c++有84个,python有33个,java有53个. 差异1:go语言不允许隐式转换,别名和原有类型也不能进行隐 ...
 - 基于Linux下的Ubuntu操作系统常用命令
			
一 .linux操作系统的特点 1.linux下一切皆文件 2.linux系统就像一个倒置数 3.linux系统支持多用户.多任务 二. Ubuntu --"乌班图"操作系统 Ub ...
 - MSIC总结取证分析——日志分析
			
MSIC总结取证分析 一.日志分析: 1.常见日志分析类型: 2.常见一些考点: (1)还原特定IP攻击手段(SQL注入.暴力破解.命令执行等),或还原最初攻击时间: (2)寻找flag或者特定文件解 ...
 - ionic+vue+capacitor系列笔记--03项目使用Native插件
			
话不多说,直接上代码 下载依赖 npm install @capacitor/camera 添加权限配置代码到安卓文件夹里的 AndroidManifest.xml <uses-permissi ...
 - vue打包---放到服务器下(一个服务器多个项目需要配置路径),以及哈希模式和历史模式的不同配置方法
			
哈希模式,好用,不需要服务器配合分配路径指向,自己单机就可以打开了 接下来上代码截图 接下来开始截图 历史模式 历史模式需要后端支持 打包后自己直接点击是打不开的 截图如下
 - 系列化和反序列化的概述-对象的序列化_Object Output Stream类
			
系列化和反序列化的概述 Java提供了一种对象序列化的机制.用一个字节序列可以表示一个对象,该字节序列包含该对象的数据对象的类型和对象中存储的属性等信息.字节序列写出到文件之后,相当于文件中持久保存了 ...
 - inline的作用
			
1:inline可以跳过调用,直接引用,类似与直接将函数中的代码拿到当前函数中一样 2:在.h中函数重复的时候可以用inline来解决冲突问题
 - 【C++ 泛型编程01:模板】函数模板与类模板
			
[模板] 除了OOP外,C++另一种编程思想称为 泛型编程 ,主要利用的技术就是模板 C++提供两种模板机制:函数模板和类模板 函数模板 函数模板作用 建立一个通用函数,其函数返回值类型和形参类型可以 ...
 - drf-day8——断点调试、认证.权限.频率的源码分析、基于APIView编写分页、全局异常处理
			
目录 一.断点调试使用 二.认证,权限,频率源码分析(了解) 2.1 权限类的执行源码 2.2 认证源码分析 2.3 频率源码分析 2.4 自定义频率类(了解) 2.5 SimpleRateThrot ...
 - @RestControllerAdvice全局异常统一处理
			
spring项目中,我们通常规定了返回的格式(成功-失败-异常),特别是异常怎么处理方便呢? 1.自定义状态码实体 package com.ruoyi.common.constant; /** * 返 ...