当前位置: 首页 > news >正文

[100天算法】-不同路径 III(day 73)

题目描述

在二维网格 grid 上,有 4 种类型的方格:1 表示起始方格。且只有一个起始方格。
2 表示结束方格,且只有一个结束方格。
0 表示我们可以走过的空方格。
-1 表示我们无法跨越的障碍。
返回在四个方向(上、下、左、右)上行走时,从起始方格到结束方格的不同路径的数目。每一个无障碍方格都要通过一次,但是一条路径中不能重复通过同一个方格。示例 1:输入:[[1,0,0,0],[0,0,0,0],[0,0,2,-1]]
输出:2
解释:我们有以下两条路径:
1. (0,0),(0,1),(0,2),(0,3),(1,3),(1,2),(1,1),(1,0),(2,0),(2,1),(2,2)
2. (0,0),(1,0),(2,0),(2,1),(1,1),(0,1),(0,2),(0,3),(1,3),(1,2),(2,2)
示例 2:输入:[[1,0,0,0],[0,0,0,0],[0,0,0,2]]
输出:4
解释:我们有以下四条路径:
1. (0,0),(0,1),(0,2),(0,3),(1,3),(1,2),(1,1),(1,0),(2,0),(2,1),(2,2),(2,3)
2. (0,0),(0,1),(1,1),(1,0),(2,0),(2,1),(2,2),(1,2),(0,2),(0,3),(1,3),(2,3)
3. (0,0),(1,0),(2,0),(2,1),(2,2),(1,2),(1,1),(0,1),(0,2),(0,3),(1,3),(2,3)
4. (0,0),(1,0),(2,0),(2,1),(1,1),(0,1),(0,2),(0,3),(1,3),(1,2),(2,2),(2,3)
示例 3:输入:[[0,1],[2,0]]
输出:0
解释:
没有一条路能完全穿过每一个空的方格一次。
请注意,起始和结束方格可以位于网格中的任意位置。提示:1 <= grid.length * grid[0].length <= 20
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/unique-paths-iii
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路

从起始格子开始,尝试每一个 0 空格。当走到 2 时,如果此时网格没有还没走过的空格,说明这是一条可行的路径。也就是说我们需要用一个方式来标志已经走过的空格,可以把格子设为 -1,回溯时需要把格子重新设置为 0,不影响其他路径的尝试。

当我们走到 2 时,如何判断网格中是否还有未走过的空格?

每次都去遍历整个网格的话,时间复杂度太高。我们可以在开始先统计网格中一共有多少个可以走的格子,每走过一个格子计数器就减一。

复杂度

  • 时间复杂度:$O(4^{mn})$, m, n 分别是网格的长宽。找到起始格子和统计空格用了 $O(mn)$,递归的时间复杂度 $O(4^{mn})$,网格一共有 $mn$ 个格子,每个格子有 4 个方向可以走。
  • 空间复杂度:递归栈的最大空间 O(m∗n)。

p.s. 下方代码是我看错题了,求了所有路径。实际上只需要一个计数器来记录路径数,不消耗额外空间。

代码

JavaScript Code

/*** @param {number[][]} grid* @return {number}*/
var uniquePathsIII = function (grid) {const offsets = [[-1, 0],[1, 0],[0, -1],[0, 1],];const ans = [];const dfs = (grid, x, y, spaceCnt, path) => {if (x < 0 || x >= grid.length || y < 0 || y >= grid[0].length) return;if (grid[x][y] === 2) {spaceCnt === 0 && ans.push([...path]);return;}if (grid[x][y] === -1) return;grid[x][y] = -1; // mark// recursionfor (const [ox, oy] of offsets) {// p.s. 如果 (x+ox, y+oy) 不在网格中或者是障碍的话,也可以提前剪枝。dfs(grid, x + ox, y + oy, spaceCnt - 1, [...path, [x, y]]);}grid[x][y] = 0; // backtrack};let startPos = {};const init = grid => {let spaceCnt = 1; // 起始方格也是要走的一个格子for (let x = 0; x < grid.length; x++) {for (let y = 0; y < grid[x].length; y++) {if (grid[x][y] === 1) startPos = { x, y };if (grid[x][y] === 0) spaceCnt++;}}return spaceCnt;};// 统计要走的格子总数const spaceCnt = init(grid);dfs(grid, startPos.x, startPos.y, spaceCnt, []);return ans.length;
};

相关文章:

  • 计算机杂谈系列精讲100篇-【自动化控制】PLC
  • Android通信安全之HTTPS
  • 案例续集留言板
  • 基于python+TensorFlow+Django卷积网络算法+深度学习模型+蔬菜识别系统
  • 数据分析面试题1
  • git分支与tag标签的介绍与使用)
  • 通信信道:无线信道中衰落的类型和分类
  • java Stream编程笔记
  • Accelerate 0.24.0文档 一:极速入门
  • springboot国际化
  • Java设计模式之访问者模式
  • [笔记]深入解析Windows操作系统《番外》windows关键进程解释
  • Module build failed (from ./node_modules/postcss-loader/src/index.js):
  • Android 10.0 系统内存优化之修改dalvik虚拟机的内存参数
  • JavaScript将去掉小数多余的0,不用正则匹配,简单易懂
  • [nginx文档翻译系列] 控制nginx
  • 【跃迁之路】【444天】程序员高效学习方法论探索系列(实验阶段201-2018.04.25)...
  • Bootstrap JS插件Alert源码分析
  • bootstrap创建登录注册页面
  • JavaScript工作原理(五):深入了解WebSockets,HTTP/2和SSE,以及如何选择
  • Js基础知识(一) - 变量
  • oschina
  • PHP面试之三:MySQL数据库
  • React组件设计模式(一)
  • Vue--数据传输
  • Vue学习第二天
  • 初识MongoDB分片
  • 开源SQL-on-Hadoop系统一览
  • 离散点最小(凸)包围边界查找
  • 那些被忽略的 JavaScript 数组方法细节
  • 七牛云假注销小指南
  • 深入浅出Node.js
  • 世界编程语言排行榜2008年06月(ActionScript 挺进20强)
  • 收藏好这篇,别再只说“数据劫持”了
  • 说说动画卡顿的解决方案
  • 通过npm或yarn自动生成vue组件
  • 微信小程序上拉加载:onReachBottom详解+设置触发距离
  • 微信小程序实战练习(仿五洲到家微信版)
  • 写给高年级小学生看的《Bash 指南》
  • 应用生命周期终极 DevOps 工具包
  • 阿里云重庆大学大数据训练营落地分享
  • ​3ds Max插件CG MAGIC图形板块为您提升线条效率!
  • ​html.parser --- 简单的 HTML 和 XHTML 解析器​
  • ​LeetCode解法汇总307. 区域和检索 - 数组可修改
  • $L^p$ 调和函数恒为零
  • (C语言)字符分类函数
  • (Java)【深基9.例1】选举学生会
  • (二)Pytorch快速搭建神经网络模型实现气温预测回归(代码+详细注解)
  • (附源码)springboot美食分享系统 毕业设计 612231
  • (一一四)第九章编程练习
  • (转)原始图像数据和PDF中的图像数据
  • ***微信公众号支付+微信H5支付+微信扫码支付+小程序支付+APP微信支付解决方案总结...
  • .net core 3.0 linux,.NET Core 3.0 的新增功能
  • .NET Remoting学习笔记(三)信道
  • .NET 编写一个可以异步等待循环中任何一个部分的 Awaiter