问题描述

403. 青蛙过河 (Hard)

一只青蛙想要过河。 假定河流被等分为若干个单元格,并且在每一个单元格内都有可能放有一块石子(也有可能没有)。

青蛙可以跳上石子,但是不可以跳入水中。

给你石子的位置列表 stones(用单元格序号 升序 表示),

请判定青蛙能否成功过河(即能否在最后一步跳至最后一块石子上)。开始时,

青蛙默认已站在第一块石子上,并可以假定它第一步只能跳跃 1 个单位(即只能从单元格 1 跳至单元格 2 )。

如果青蛙上一步跳跃了 k 个单位,那么它接下来的跳跃距离只能选择为 k - 1kk + 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 <= 2000
  • 0 <= stones[i] <= 2³¹ - 1
  • stones[0] == 0
  • stones 按严格升序排列

解题思路

记忆化搜索

我们考虑dfs(i, k)表示从第i个石子跳k步,之后能否到达终点;

  • 如果从第i个石子跳k步,不能到达另一个石子,return false;
  • 否则,记从第i个石子跳k步到达的新石头的索引为new_idx,那么只要从new_idxk + 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)的更多相关文章

  1. Java实现 LeetCode 403 青蛙过河

    403. 青蛙过河 一只青蛙想要过河. 假定河流被等分为 x 个单元格,并且在每一个单元格内都有可能放有一石子(也有可能没有). 青蛙可以跳上石头,但是不可以跳入水中. 给定石子的位置列表(用单元格序 ...

  2. [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 ...

  3. NOIP 2005 青蛙过河

    做题记录:2016-08-10 21:58:09 题目描述 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都 ...

  4. ooj 1066 青蛙过河DP

    http://121.249.217.157/JudgeOnline/problem.php?id=1066 1066: 青蛙过河 时间限制: 1 Sec  内存限制: 64 MB提交: 58  解决 ...

  5. 趣味算法——青蛙过河(JAVA)

    青蛙过河是一个非常有趣的智力游戏,其大意如下: 一条河之间有若干个石块间隔,有两队青蛙在过河,每队有3只青蛙,这些青蛙只能向前移动,不能向后移动,且一次只能有一只青蛙向前移动.在移动过程中,青蛙可以向 ...

  6. HRBUST 1186 青蛙过河 (思路错了)

    在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙可能到达的点看成数轴上的一串 ...

  7. P1244 青蛙过河

    P1244 青蛙过河NOI2000主要思想:数学归纳法 递推 压位高精度 化归 理解能力和找规律的能力题意再述:1.青蛙从上到下必须连续递增或者下面是石墩 而不能是1 12 33 4而且每时每刻都要满 ...

  8. 洛谷P1244 青蛙过河

    P1244 青蛙过河 362通过 525提交 题目提供者该用户不存在 标签 难度普及- 时空限制1s / 128MB 提交 讨论 题解 最新讨论更多讨论 题目什么意思 题目看不懂啊 题目描述 有一条河 ...

  9. 洛谷 P1244 青蛙过河

    P1244 青蛙过河 题目描述 有一条河,左边一个石墩(A区)上有编号为1,2,3,4,…,n的n只青蛙,河中有k个荷叶(C区),还有h个石墩(D区),右边有一个石墩(B区),如下图所示.n只青蛙要过 ...

  10. 洛谷P1244 [NOI2000] 青蛙过河 [2017年4月计划 动态规划07]

    P1244 青蛙过河 题目描述 有一条河,左边一个石墩(A区)上有编号为1,2,3,4,…,n的n只青蛙,河中有k个荷叶(C区),还有h个石墩(D区),右边有一个石墩(B区),如下图所示.n只青蛙要过 ...

随机推荐

  1. 第四篇:前端之BOM与DOM

    前端基础之BOM和DOM   前戏 到目前为止,我们已经学过了JavaScript的一些简单的语法.但是这些简单的语法,并没有和浏览器有任何交互. 也就是我们还不能制作一些我们经常看到的网页的一些交互 ...

  2. 12、synchronized和Lock的使用

    转载自 1.多并发案例: 一个车站有三个窗口同时卖30张票,每个窗口都有40个人在排队买票,在多线程情况下,不加锁,线程不安全,导致卖票不准确 package com.example.Lock; /* ...

  3. [R语言] 基于R语言实现树形图的绘制

    树状图(或树形图)是一种网络结构.它由一个根节点组成,根节点产生由边或分支连接的多个节点.层次结构的最后一个节点称为叶.本文主要基于R语言实现树形图的绘制.关于python实现树形图的绘制见:基于ma ...

  4. 《Effective C++》实现 章节

    Item26:尽可能延后变量定义式的出现时间 Item27:尽量少做转型动作 关于这一点,专门开了一个新的总结: http://blog.csdn.net/m0_37316917/article/de ...

  5. P8622 [蓝桥杯 2014 国 B] 生物芯片

    简要题意 有 \(N\) 个二进制数,编号为 \(1\sim N\),初始时都是 \(0\).博士进行了 \(N-1\) 次操作,在第 \(i\) 次操作时,会将 \(1\sim N\) 中所有编号为 ...

  6. appium如何连接多台设备

    我们在做app自动化的时候,若要考虑兼容性问题,需要跑几台设备,要是一台一台的跑比较耗 时,因此需要考虑使用多线程来同时操作多台设备. 1.我们拿两台设备来模拟操作下,使用:adb devices查看 ...

  7. 超级容易理解的函数节流(throttle)

    今天搞了一个简单的写法 话不多说,直接上代码 <!DOCTYPE html> <html lang="en"> <head> <meta ...

  8. hashlib加密模块及subprocess远程命令模块

    hashlib加密模块及subprocess远程命令模块 一.hashlib加密模块 1.加密模块简介 1.加密模块简介 将明文数据进行加密处理,转变为密文数据再存储或者传输,这样的安全机制可以让用户 ...

  9. sync.Once 使用及解析

    目录 前言 1. sync.Once 简介 2. sync.Once 源码解析 2.1 为什么 done 作为第一个字段 2.2 Do 方法的实现细节 2.3 其他重要细节 3. sync.Once ...

  10. python70 前端框架之vue js的集中循环方式、key值的解释、input事件、v-model双向数据绑定、过滤案例、事件修饰符、按键修饰符、表单控制

    js的几种循环方式 v-for可以循环的变量 可以循环的: 数组.数组带索引 对象.对象带key.value 字符串 字符串带索引 数字.数字带索引 <!DOCTYPE html> < ...