龗孖 龗孖
首页
  • JAVA
  • 设计模式
  • 前端文章

    • JavaScript
  • 学习笔记

    • 《JavaScript教程》
    • 《JavaScript高级程序设计》
    • 《ES6 教程》
    • 《Vue》
    • 《React》
    • 《TypeScript 从零实现 axios》
    • 《Git》
    • TypeScript
    • JS设计模式总结
  • 页面

    • HTML
    • CSS
  • 技术文档
  • GitHub技巧
  • Nodejs
  • 博客搭建
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
关于
收藏
  • 分类
  • 标签
  • 归档
GitHub (opens new window)

靇孖

某微型企业非牛逼技术专家。
首页
  • JAVA
  • 设计模式
  • 前端文章

    • JavaScript
  • 学习笔记

    • 《JavaScript教程》
    • 《JavaScript高级程序设计》
    • 《ES6 教程》
    • 《Vue》
    • 《React》
    • 《TypeScript 从零实现 axios》
    • 《Git》
    • TypeScript
    • JS设计模式总结
  • 页面

    • HTML
    • CSS
  • 技术文档
  • GitHub技巧
  • Nodejs
  • 博客搭建
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
关于
收藏
  • 分类
  • 标签
  • 归档
GitHub (opens new window)
  • JAVA

  • MQ

  • 工具

  • 微服务

  • 数据库

  • 其他

  • 程序设计

  • 算法

    • 限流算法
    • 数组中重复的数字
    • 二维数组中的查找
    • 替换空格
    • 从尾到头打印链表
    • 重建二叉树
    • 二叉树的下一个结点
    • 用两个栈实现队列
    • 4 变态跳台阶
    • 旋转数组的最小数字
    • 矩阵中的路径
      • 题目描述
      • 解题思路
    • 机器人的运动范围
    • 剪绳子
    • 二进制中 1 的个数
    • 数值的整数次方
    • 打印从 1 到最大的 n 位数
    • 2 删除链表中重复的结点
    • 正则表达式匹配
    • 表示数值的字符串
    • 调整数组顺序使奇数位于偶数前面
    • 链表中倒数第 K 个结点
    • 链表中环的入口结点
    • 反转链表
    • 合并两个排序的链表
    • 树的子结构
    • 二叉树的镜像
    • 对称的二叉树
    • 顺时针打印矩阵
    • 包含 min 函数的栈
    • 栈的压入、弹出序列
    • 3 按之字形顺序打印二叉树
    • 二叉搜索树的后序遍历序列
    • 二叉树中和为某一值的路径
    • 复杂链表的复制
    • 二叉搜索树与双向链表
    • 序列化二叉树
    • 字符串的排列
    • 数组中出现次数超过一半的数字
    • 最小的 K 个数
    • 2 字符流中第一个不重复的字符
    • 连续子数组的最大和
    • 从 1 到 n 整数中 1 出现的次数
    • 数字序列中的某一位数字
    • 把数组排成最小的数
    • 把数字翻译成字符串
    • 礼物的最大价值
    • 最长不含重复字符的子字符串
    • 丑数
    • 第一个只出现一次的字符位置
    • 数组中的逆序对
    • 两个链表的第一个公共结点
    • 数字在排序数组中出现的次数
    • 二叉查找树的第 K 个结点
    • 2 平衡二叉树
    • 数组中只出现一次的数字
    • 2 和为 S 的连续正数序列
    • 2 左旋转字符串
    • 滑动窗口的最大值
    • n 个骰子的点数
    • 扑克牌顺子
    • 圆圈中最后剩下的数
    • 股票的最大利润
    • 求 1+2+3+...+n
    • 不用加减乘除做加法
    • 构建乘积数组
    • 把字符串转换成整数
    • 树中两个节点的最低公共祖先
  • 服务端
  • 算法
龗孖
2022-10-12
目录

矩阵中的路径

# 12. 矩阵中的路径

牛客网 (opens new window)

# 题目描述

判断在一个矩阵中是否存在一条包含某字符串所有字符的路径。路径可以从矩阵中的任意一个格子开始,每一步可以在矩阵中向上下左右移动一个格子。如果一条路径经过了矩阵中的某一个格子,则该路径不能再进入该格子。

例如下面的矩阵包含了一条 bfce 路径。


# 解题思路

使用回溯法(backtracking)进行求解,它是一种暴力搜索方法,通过搜索所有可能的结果来求解问题。回溯法在一次搜索结束时需要进行回溯(回退),将这一次搜索过程中设置的状态进行清除,从而开始一次新的搜索过程。例如下图示例中,从 f 开始,下一步有 4 种搜索可能,如果先搜索 b,需要将 b 标记为已经使用,防止重复使用。在这一次搜索结束之后,需要将 b 的已经使用状态清除,并搜索 c。


本题的输入是数组而不是矩阵(二维数组),因此需要先将数组转换成矩阵。

public class Solution {
    private final static int[][] next = {{0, -1}, {0, 1}, {-1, 0}, {1, 0}};
    private int rows;
    private int cols;

    public boolean hasPath (String val, int rows, int cols, String path) {
        if (rows == 0 || cols == 0) return false;
        this.rows = rows;
        this.cols = cols;
        char[] array = val.toCharArray();
        char[][] matrix = buildMatrix(array);
        char[] pathList = path.toCharArray();
        boolean[][] marked = new boolean[rows][cols];
        for (int i = 0; i < rows; i++)
            for (int j = 0; j < cols; j++)
                if (backtracking(matrix, pathList, marked, 0, i, j))
                    return true;

        return false;
    }

    private boolean backtracking(char[][] matrix, char[] pathList,
                                 boolean[][] marked, int pathLen, int r, int c) {

        if (pathLen == pathList.length) return true;
        if (r < 0 || r >= rows || c < 0 || c >= cols
                || matrix[r][c] != pathList[pathLen] || marked[r][c]) {

            return false;
        }
        marked[r][c] = true;
        for (int[] n : next)
            if (backtracking(matrix, pathList, marked, pathLen + 1, r + n[0], c + n[1]))
                return true;
        marked[r][c] = false;
        return false;
    }

    private char[][] buildMatrix(char[] array) {
        char[][] matrix = new char[rows][cols];
        for (int r = 0, idx = 0; r < rows; r++)
            for (int c = 0; c < cols; c++)
                matrix[r][c] = array[idx++];
        return matrix;
    }

    public static void main(String[] args) {
        Solution solution = new Solution();
        String val = "ABCESFCSADEE";
        int rows = 3;
        int cols = 4;
        String path = "ABCCED";
        boolean res = solution.hasPath(val, rows, cols, path);
        System.out.println(res);
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
上次更新: 2024/11/03, 21:06:16
旋转数组的最小数字
机器人的运动范围

← 旋转数组的最小数字 机器人的运动范围→

最近更新
01
树中两个节点的最低公共祖先
10-17
02
hexo多平台多博客网站同步
09-04
03
最长不含重复字符的子字符串
09-03
更多文章>
Theme by Vdoing | Copyright © 2015-2024 Ling ma | 996.icu | 京ICP备16011424号-1
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式