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

LeetCode-79. 单词搜索【数组 字符串 回溯 矩阵】

LeetCode-79. 单词搜索【数组 字符串 回溯 矩阵】

  • 题目描述:
  • 解题思路一:回溯 回溯三部曲。这里比较关键的是给board做标记,防止之后搜索时重复访问。
  • 解题思路二:回溯算法 + dfs,直接看代码,很容易理解。visited哈希,防止重复访问。
  • 解题思路三:0

题目描述:

给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。

单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许被重复使用。

示例 1:
在这里插入图片描述
输入:board = [[“A”,“B”,“C”,“E”],[“S”,“F”,“C”,“S”],[“A”,“D”,“E”,“E”]], word = “ABCCED”
输出:true
示例 2:
在这里插入图片描述
输入:board = [[“A”,“B”,“C”,“E”],[“S”,“F”,“C”,“S”],[“A”,“D”,“E”,“E”]], word = “SEE”
输出:true
示例 3:
在这里插入图片描述
输入:board = [[“A”,“B”,“C”,“E”],[“S”,“F”,“C”,“S”],[“A”,“D”,“E”,“E”]], word = “ABCB”
输出:false

提示:

m == board.length
n = board[i].length
1 <= m, n <= 6
1 <= word.length <= 15
board 和 word 仅由大小写英文字母组成

进阶:你可以使用搜索剪枝的技术来优化解决方案,使其在 board 更大的情况下可以更快解决问题?

解题思路一:回溯 回溯三部曲。这里比较关键的是给board做标记,防止之后搜索时重复访问。

  1. 递归函数参数
    这里的参数是:
    当前元素在矩阵 board 中的行列索引 i 和 j ,当前目标字符在 word 中的索引 index 。

  2. 递归终止条件
    返回 false: (1) 行或列索引越界 或 (2) 当前矩阵元素与目标字符不同 或 (3) 当前矩阵元素已访问过 ( (3) 可合并至 (2) ) 。
    返回 true : k = len(word) - 1 ,即字符串 word 已全部匹配。

  3. 单层搜索的逻辑
    标记当前矩阵元素: 将 board[i][j] 修改为 空字符 ‘’ ,代表此元素已访问过,防止之后搜索时重复访问。
    搜索下一单元格: 朝当前元素的 上、下、左、右 四个方向开启下层递归,使用 或 连接 (代表只需找到一条可行路径就直接返回,不再做后续 DFS ),并记录结果至 res 。
    还原当前矩阵元素: 将 board[i][j] 元素还原至初始值,即 word[k] 。

  4. 返回值: 返回布尔量 res ,代表是否搜索到目标字符串。

使用空字符(Python: ‘’ , Java/C++: ‘\0’ )做标记是为了防止标记字符与矩阵原有字符重复。当存在重复时,此算法会将矩阵原有字符认作标记字符,从而出现错误。
在这里插入图片描述

class Solution:def __init__(self):self.dirs = [(-1, 0), (0, 1), (1, 0), (0, -1)]def exist(self, board: List[List[str]], word: str) -> bool:m, n = len(board), len(board[0])for i in range(m):for j in range(n):if self.backtracking(board, i, j, 0, word):return Truereturn Falsedef backtracking(self, board, x, y, index, word):if x < 0 or x >= len(board) or y < 0 or y >= len(board[0]) or board[x][y] != word[index]:return Falseif index == len(word) - 1:return Trueres = Falsefor d in self.dirs:nextx = x + d[0]nexty = y + d[1]board[x][y] = ''res = self.backtracking(board, nextx, nexty, index+1, word) or resboard[x][y] = word[index]return res

在代码中,M,N 分别为矩阵行列大小, K 为字符串 word 长度。

时间复杂度 O(3KMN): 最差情况下,需要遍历矩阵中长度为 K 字符串的所有方案,时间复杂度为 O(3K);矩阵中共有 MN 个起点,时间复杂度为 O(MN) 。
方案数计算: 设字符串长度为 K ,搜索中每个字符有上、下、左、右四个方向可以选择,舍弃回头(上个字符)的方向,剩下 333 种选择,因此方案数的复杂度为 O(3K)。
空间复杂度 O(K) : 搜索过程中的递归深度不超过 K ,因此系统因函数调用累计使用的栈空间占用 O(K) (因为函数返回后,系统调用的栈空间会释放)。最坏情况下 K=MN,递归深度为 MN ,此时系统栈使用 O(MN) 的额外空间。

解题思路二:回溯算法 + dfs,直接看代码,很容易理解。visited哈希,防止重复访问。

class Solution:def exist(self, board: List[List[str]], word: str) -> bool:row = len(board)col = len(board[0])def helper(i, j, k, visited):#print(i,j, k,visited)if k == len(word):return Truefor x, y in [(-1, 0), (1, 0), (0, 1), (0, -1)]:tmp_i = x + itmp_j = y + jif 0 <= tmp_i < row and 0 <= tmp_j < col and (tmp_i, tmp_j) not in visited \and board[tmp_i][tmp_j] == word[k]:visited.add((tmp_i, tmp_j))if helper(tmp_i, tmp_j, k+1, visited):return Truevisited.remove((tmp_i, tmp_j)) # 回溯return Falsefor i in range(row):for j in range(col):if board[i][j] == word[0] and helper(i, j, 1,{(i, j)}) :return Truereturn False

时间复杂度:O(3KMN)
空间复杂度:O(K)

解题思路三:0


时间复杂度:O(n)
空间复杂度:O(n)

http://www.lryc.cn/news/332711.html

相关文章:

  • 游戏引擎之高级动画技术
  • Oracle 数据库中的全文搜索
  • 代码随想录阅读笔记-二叉树【二叉搜索树中的众数】
  • AcWing-游戏
  • Mybatis——一对一映射
  • Web 安全之 SSL 剥离攻击详解
  • 数据结构——顺序表(C语言)
  • 利用Idea实现Ajax登录(maven工程)
  • 环信IM集成教程——Web端UIKit快速集成与消息发送
  • Anaconda如何切换国内镜像源
  • Android 14.0 添加自定义服务,并生成jar给第三方app调用
  • 解决沁恒ch592单片机在tmos中使用USB总线时,接入USB Hub无法枚举频繁Reset的问题
  • nvm保姆级安装使用教程
  • 大语言模型LLM《提示词工程指南》学习笔记02
  • 【realme x2手机解锁BootLoader(简称BL)】
  • 攻防世界 wife_wife
  • Visual Studio安装下载进度为零已解决
  • 矩阵空间秩1矩阵小世界图
  • 《QT实用小工具·十三》FlatUI辅助类之各种炫酷的控件集合
  • dm8 备份与恢复
  • Vue项目中引入html页面(vue.js中引入echarts数据大屏html [静态非数据传递!] )
  • ASTM C1186-22 纤维水泥平板
  • NoSQL概述
  • 爬虫实战一、Scrapy开发环境(Win10+Anaconda3)搭建
  • llama.cpp运行qwen0.5B
  • 【接口】HTTP(3) |GET和POST两种基本请求方法有什么区别
  • 金陵科技学院软件工程学院软件工程专业
  • Android 关于apk反编译d2j-dex2jar classes.dex失败的几种方法
  • Django--admin 后台管理站点
  • JavaScript(六)---【回调、异步、promise、Async】