Leetcode.1559 二维网格图中探测环

news2025/1/11 20:09:13

题目链接

Leetcode.1559 二维网格图中探测环 rating : 1838

题目描述

给你一个二维字符网格数组 g r i d grid grid ,大小为 m x n ,你需要检查 g r i d grid grid 中是否存在 相同值 形成的环。

一个环是一条开始和结束于同一个格子的长度 大于等于 4 4 4 的路径。对于一个给定的格子,你可以移动到它上、下、左、右四个方向相邻的格子之一,可以移动的前提是这两个格子有 相同的值

同时,你也不能回到上一次移动时所在的格子。比方说,环 ( 1 , 1 ) − > ( 1 , 2 ) − > ( 1 , 1 ) (1, 1) -> (1, 2) -> (1, 1) (1,1)>(1,2)>(1,1) 是不合法的,因为从 ( 1 , 2 ) (1, 2) (1,2) 移动到 ( 1 , 1 ) (1, 1) (1,1) 回到了上一次移动时的格子。

如果 g r i d grid grid 中有相同值形成的环,请你返回 true ,否则返回 false

示例 1:

在这里插入图片描述

输入:grid = [[“a”,“a”,“a”,“a”],[“a”,“b”,“b”,“a”],[“a”,“b”,“b”,“a”],[“a”,“a”,“a”,“a”]]
输出:true
解释:如下图所示,有 2 个用不同颜色标出来的环:
在这里插入图片描述

示例 2:

在这里插入图片描述

输入:grid = [[“c”,“c”,“c”,“a”],[“c”,“d”,“c”,“c”],[“c”,“c”,“e”,“c”],[“f”,“c”,“c”,“c”]]
输出:true
解释:如下图所示,只有高亮所示的一个合法环:
在这里插入图片描述

示例 3:

在这里插入图片描述

输入:grid = [[“a”,“b”,“b”],[“b”,“z”,“b”],[“b”,“b”,“a”]]
输出:false

提示:

  • m = g r i d . l e n g t h m = grid.length m=grid.length
  • n = g r i d [ i ] . l e n g t h n = grid[i].length n=grid[i].length
  • 1 ≤ m ≤ 500 1 \leq m \leq 500 1m500
  • 1 ≤ n ≤ 500 1 \leq n \leq 500 1n500
  • g r i d grid grid 只包含小写英文字母。

解法一:并查集

我们从左上角开始遍历,假设当前位置是 ( i , j ) (i,j) (i,j),我们只用判断它的右边 和 下面的位置即 ( i + 1 , j ) (i+1,j) (i+1,j) ( i , j + 1 ) (i,j + 1) (i,j+1)

如果 g r i d [ i ] [ j ] = g r i d [ i + 1 ] [ j ] grid[i][j] = grid[i + 1][j] grid[i][j]=grid[i+1][j]

  • g r i d [ i ] [ j ] grid[i][j] grid[i][j] g r i d [ i + 1 ] [ j ] grid[i+1][j] grid[i+1][j] 处于两个连通块,那么我就他们合并到一起;
  • 如果 g r i d [ i ] [ j ] grid[i][j] grid[i][j] g r i d [ i + 1 ] [ j ] grid[i+1][j] grid[i+1][j] 已经处于同一个连通块了,那么说明存在环,直接返回 true

如果 g r i d [ i ] [ j ] = g r i d [ i ] [ j + 1 ] grid[i][j] = grid[i ][j + 1] grid[i][j]=grid[i][j+1]

  • g r i d [ i ] [ j ] grid[i][j] grid[i][j] g r i d [ i ] [ j + 1 ] grid[i][j + 1] grid[i][j+1] 处于两个连通块,那么我就他们合并到一起;
  • 如果 g r i d [ i ] [ j ] grid[i][j] grid[i][j] g r i d [ i ] [ j + 1 ] grid[i][j+1] grid[i][j+1] 已经处于同一个连通块了,那么说明存在环,直接返回 true

时间复杂度: O ( m × n ) O(m \times n) O(m×n)

C++代码:

class Solution {
public:
    bool containsCycle(vector<vector<char>>& grid) {
        int m = grid.size() , n = grid[0].size() , len = m * n;
        vector<int> p(len);

        for(int i = 0;i < len;i++) p[i] = i;

        function<int(int)> find = [&](int x) -> int{
            if(x != p[x]){
                p[x] = find(p[x]);
            }
            return p[x];
        };

        auto is_connected = [&](int a,int b){
            int x = find(a) , y = find(b);
            if(x == y) return true;
            return false;
        };

        auto merge = [&](int a,int b){
            int x = find(a) , y = find(b);
            p[x] = y;
        };

        for(int i = 0;i < m;i++){
            for(int j = 0;j < n;j++){
                int x = i * n + j , y;

                if(j + 1 < n && grid[i][j] == grid[i][j + 1]){
                    y = i * n + j + 1;
                    if(is_connected(x,y)) return true;
                    merge(x,y);
                }

                if(i + 1 < m && grid[i][j] == grid[i + 1][j]){
                    y = (i + 1) * n + j;
                    if(is_connected(x,y)) return true;
                    merge(x,y);
                }
            }
        }
        return false;
    }
};

解法二:dfs

我们每次从没有访问过的位置出发开始超 上下左右 四个方向开始访问。

访问的时候注意,不能沿着来的方向再反方向回去。

在 dfs 的过程中,如果我们访问到了跟当前位置值相同,并且之前访问过的点,就说明存在环,直接返回 true

否则说明不存在环,最后返回 false

时间复杂度: O ( m × n ) O(m \times n) O(m×n)

C++代码:

//(-1,0) (0,-1) (0,1) (1,0) 分别为 上左右下
//这样对于 k 那么它的反方向就是 3 - k

const int dx[4] = {-1,0,0,1};
const int dy[4] = {0,-1,1,0};
class Solution {
public:
    bool containsCycle(vector<vector<char>>& grid) {
        int m = grid.size() , n = grid[0].size();

        bool vis[m][n];
        memset(vis,false,sizeof vis);

        function<bool(int,int,int)> dfs = [&](int x,int y,int dir) ->bool{
            for(int k = 0;k < 4;k++){
                int nx = x + dx[k] , ny = y + dy[k];
                //dir == k 说明现在的方向 跟 来的时候的方向 相反
                //我们不能再沿着来的路回去,所以这里直接跳过
                if(dir == k || nx < 0 || nx >= m || ny < 0 || ny >= n || grid[x][y] != grid[nx][ny]) continue;
                //此时 (nx,ny) 这个点之前已经访问过了 说明存在环 直接返回true
                if(vis[nx][ny]) return true;
                vis[nx][ny] = true;

                if(dfs(nx,ny,3 - k)) return true;
            }

            return false;
        };

        for(int i = 0;i < m;i++){
            for(int j = 0;j < n;j++){
                if(vis[i][j]) continue;
                vis[i][j] = true;
                if(dfs(i,j,-1)) return true;
            }
        }

        return false;
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/855428.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java基础(四)

循环结构 1. while循环 循环变量初始化 while (循环条件){循环体; }循环条件是一个布尔类型的表达式&#xff0c;他的值为 true 时执行循环体&#xff0c;如果为 false 时终止循环体。while 循环是先判断条件是否成立&#xff0c;再决定是否执行循环体。如果第一次循环时&…

【APITable】教程:创建并运行一个自建小程序

1.进入APITable&#xff0c;在想要创建小程序的看板页面点击右上角的【小程序】&#xff0c;进入小程序编辑页面。 2.创建一个新的小程序区。 点击【 添加小程序】 点击创建小程序&#xff0c;选择模板&#xff0c;输入名字。 3.确定后进入小程序部署引导页面。 4.打开Xshell 7…

06-3_Qt 5.9 C++开发指南_多窗体应用程序的设计(主要的窗体类及其用途;窗体类重要特性设置;多窗口应用程序设计)

文章目录 1. 主要的窗体类及其用途2. 窗体类重要特性的设置2.1 setAttribute()函数2.2 setWindowFlags()函数2.3 setWindowState()函数2.4 setWindowModality()函数2.5 setWindowOpacity()函数 3. 多窗口应用程序设计3.1 主窗口设计3.2 QFormDoc类的设计3.3 QFormDoc类的使用3.…

Win10语言设置 - 显示语言和应用语言

前言 Win10的语言设置可以设置显示语言和应用语言。其中&#xff0c;显示语言用于显示系统文字&#xff1b;应用语言用于应用程序显示文字。下文介绍如何设置。 显示语言 打开系统设置&#xff0c;选择时间和语言&#xff0c;如下图&#xff1a; 修改Windows显示语言即可更…

开源数据库Mysql_DBA运维实战 (名词解释)

SQL&#xff08;Structured Query Language 即结构化查询语言&#xff09; SQL语言主要用于存取数据、查询数据、更新数据和管理关系数据库系统&#xff0c;SQL语言由IBM开发。 SQL语言分类&#xff1a; DDL语句 数据库定义语言&#xff1a;数据库、表、视图、索引、存储过程…

CentOS7安装JDK8(实操版 | 源码安装和yum安装)

&#x1f60a; 作者&#xff1a; Eric &#x1f496; 主页&#xff1a; https://blog.csdn.net/weixin_47316183?typeblog &#x1f389; 主题&#xff1a;CentOS7安装JDK8&#xff08;实操版 | 源码安装和yum安装&#xff09; ⏱️ 创作时间&#xff1a; 2023年08月04日 …

Spring Boot 的核心注解是哪个?它主要由哪几个注解组成的?

目录 一、SpringBootApplication 二、SpringBootConfiguration 三、EnableAutoConfiguration 四、ComponentScan 一、SpringBootApplication SpringBootApplication是Spring Boot框架的核心注解之一&#xff0c;它用于标识一个主配置类&#xff0c;通常是项目的入口类。该…

Titanic细节记录一

目录 chunker header index_col names Series与DataFrame的区别 df.columns del和drop的区别 reset_index loc与iloc的区别 不同的排序方式 sort_values sort_index DataFrame相加 describe函数查看数据基本信息 查看多个列的数据时使用列表 处理缺失值的几种思路 …

Adjusted frame length exceeds 1677216:16777318-discarded

问题背景 消息队列的nameser和broker运行后&#xff0c;在console无法显示&#xff0c;进入broker.log&#xff0c;namesrv.log 显示如下报错Adjusted frame length exceeds 1677216:16777318-discarded 问题原因 经定位&#xff0c;rocketmq底层集群间使用netty通信&#…

多语言海外购物商城APP系统(java开源)快速搭建

搭建一个多语言海外购物商城APP系统需要考虑以下几个方面&#xff1a;系统设计、技术架构、多语言支持和快速搭建。 一、系统设计&#xff1a; 1. 市场调研&#xff1a;了解海外购物市场的特点和需求&#xff0c;确定目标用户群体。 2. APP功能设计&#xff1a;根据市场需求&a…

【Axure高保真原型】JS版日期区间下拉选择器

今天和大家分享JS版日期区间下拉选择器的原型模板&#xff0c;该模板通过调用浏览器的下拉列表&#xff0c;所以可以获取真实的日历效果&#xff0c;具体包括哪一年二月份有29天&#xff0c;几号对应星期几&#xff0c;都是真实的。建议使用谷歌浏览器来演示&#xff0c;其他浏…

leetcode 399-除法求值

法一&#xff1a;并查集 分析示例1&#xff1a; a / b 2.0 a/ b 2.0 a/b2.0&#xff0c;说明 a 2 b a2b a2b&#xff0c; a a a和 b b b在同一个集合中 b / c 3.0 b/c3.0 b/c3.0&#xff0c;说明 b 3 c b3c b3c&#xff0c; b b b和 c c c在同一个集合中 求 a / c a/…

C++ Primer(第5版) 全书重点学习笔记

目录 第12章 动态内存 12.1 动态内存与智能指针 12.1.6 weak_ptr 12.2 动态数组 12.2.1 new和数组 12.2.2 allocator类 第12章 动态内存 12.1 动态内存与智能指针 12.1.6 weak_ptr weak_ptr是一种不控制所指向对象生存期的智能指针&#xff0c;它指向由一个shared_pt…

微信小程序申请步骤

微信公众平台链接&#xff1a;https://mp.weixin.qq.com/ 1、进到微信公众平台&#xff0c;点一下“点击注册”&#xff0c;挑选账号申请种类“小程序”&#xff0c;填好微信小程序用户信息&#xff0c;包含电子邮箱、登陆密码等。 2、微信公众平台会发送一封电子邮件&#xf…

监控Elasticsearch的关键指标

Elasticsearch 的核心职能就是对外提供搜索服务&#xff0c;所以搜索请求的吞吐和延迟是非常关键的&#xff0c;搜索是靠底层的索引实现的&#xff0c;所以索引的性能指标也非常关键&#xff0c;Elasticsearch 由一个或多个节点组成集群&#xff0c;集群自身是否健康也是需要我…

Linux网络服务之SSH

SSH 一、SSH概述1.1 定义1.2 SSH的优点1.3 OpenSSH1.3.1 定义1.3.2 SSH服务器-----sshd 二、SSH原理三、SSH登录方式3.1 方式一3.2 方式二3.3 方式三&#xff1a;跳板连接3.3.1 跳板连接概述3.3.2 具体配置 3.4 方式四&#xff1a;远程控制 四、服务端配置4.1 系统安全架构----…

企业内部wiki,让知识不再流于表面,让企业管理更加高效

企业内部wiki是一种基于wiki技术的内部知识管理平台&#xff0c;通常由企业自行搭建和维护&#xff0c;用于收集和整理企业内部的知识、经验和流程等信息。它可以帮助企业实现知识共享、协作和沟通&#xff0c;提高工作效率和团队协作能力。企业内部wiki还可以作为企业文化建设…

Python Pandas 使用示例

文章目录 使用Boolean 选择rows读取Excel表格里指定的sheet, 并跳过起始n行删除只有一个元素的行删除重复的合并多个csv文件到excel表格中获取csv文件的数据 使用Boolean 选择rows import pandas as pd# Sample DataFrame data {Name: [John, Alice, Bob, Emily],Age: [25, 3…

Segment Anything(SAM) 计算过程

给定输入图像 I ∈ R 3 H W I \in R^{3 \times H \times W} I∈R3HW。给定需要的prompts&#xff1a; M ∈ R 1 H W M \in R^{1 \times H \times W} M∈R1HW&#xff0c;代表图片的前背景信息。 P ∈ R N 2 P \in R^{N \times 2} P∈RN2&#xff0c;其中 N N N 是点的个数…

SpringBoot 底层机制分析【Tomcat 启动+Spring 容器初始化+Tomcat 如何关联Spring 容器】【下】

&#x1f600;前言 本篇博文是关于SpringBoot 底层机制分析实现&#xff0c;希望能够帮助你更好的了解SpringBoot &#x1f60a; &#x1f3e0;个人主页&#xff1a;晨犀主页 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是晨犀&#xff0c;希望我的文章可以帮助到大…