FloodFill-----洪水灌溉算法(DFS例题详解)

目录

一.图像渲染:

代码详解:

二.岛屿数量:

代码详解:

三.岛屿的最大面积:

代码详解:

四.被围绕的区域:

代码详解:

五.太平洋大西洋水流问题:

代码详解:


FloodFill算法简介:FloodFill(泛洪填充)算法是一种图像处理的基本算法,用于填充连通区域。该算法通常从一个种子点开始,沿着种子点的相邻像素进行填充,直到遇到边界或者其他指定的条件为止。FloodFill 算法的主要应用是在图像编辑软件中实现填充操作,以及在计算机图形学、计算机视觉等领域中进行区域填充。

下面我们通过一些题目来理解这个算法思想:

一.图像渲染:

  • 题目链接:733. 图像渲染 - 力扣(LeetCode)
  • 题目描述:

有一幅以 m x n 的二维整数数组表示的图画 image ,其中 image[i][j] 表示该图画的像素值大小。

你也被给予三个整数 sr ,  sc 和 newColor 。你应该从像素 image[sr][sc] 开始对图像进行 上色填充 。

为了完成 上色工作 ,从初始像素开始,记录初始坐标的 上下左右四个方向上 像素值与初始坐标相同的相连像素点,接着再记录这四个方向上符合条件的像素点与他们对应 四个方向上 像素值与初始坐标相同的相连像素点,……,重复该过程。将所有有记录的像素点的颜色值改为 newColor 。

最后返回 经过上色渲染后的图像 。 ​

  • 对应函数签名如下:

  •  思路:我们从给定的起点开始,进行深度优先搜索(上下左右四个方向)。每次搜索到一个方格时,如果其与初始位置的方格颜色相同,就将该方格的颜色更新,以防止重复搜索;如果不相同,则进行回溯。这里我们设置初始方格为target.

代码详解:

解法一:

class Solution {//记录走过的路径,防止走回头路boolean[][] used;int target;public int[][] floodFill(int[][] image, int sr, int sc, int color) {int m = image.length,n = image[0].length;used = new boolean[m][n];target = image[sr][sc];dfs(image,sr,sc,color);return image;}public void dfs(int[][] image,int i,int j,int color){int m = image.length,n = image[0].length;//剪枝,越界直接返回if(i < 0 || j < 0 || i >= m || j >= n){return ;}//使用过的位置也直接返回if(used[i][j]) return ;if(image[i][j] == target){//上下左右去深搜,符合条件的都标记为colorimage[i][j] = color;used[i][j] = true;dfs(image,i - 1,j,color);dfs(image,i + 1,j,color);dfs(image,i,j - 1,color);dfs(image,i,j + 1,color);}}
}

解法二:基于解法一,我们可以通过定义两个数组来表示方向:dx[ ],dy[ ],其中dx[ ],dy[ ]的位置要一一对应,具体操作如下:

 代码详解:

class Solution {boolean[][] used;int target;int[] dx = {-1,1,0,0};int[] dy = {0,0,1,-1};public int[][] floodFill(int[][] image, int sr, int sc, int color) {int m = image.length,n = image[0].length;used = new boolean[m][n];target = image[sr][sc];dfs(image,sr,sc,color);return image;}public void dfs(int[][] image,int i,int j,int color){int m = image.length,n = image[0].length;//每次进入都进行标记,并将该位置值改为colorused[i][j] = true;image[i][j] = color;//相当于上下左右四个方向进行深搜for(int k = 0;k < 4;k++){int x = i + dx[k],y = j + dy[k];//所有不符合条件的都不能进入深搜if(x >= 0 && x < m && y >= 0 && y < n&& !used[x][y] && image[x][y] == target){dfs(image,x,y,color);}}}
}

运行结果:

二.岛屿数量:

  • 题目链接:200. 岛屿数量 - 力扣(LeetCode)
  • 题目描述:

给你一个由 '1'(陆地)和 '0'(水)组成的的二维网格,请你计算网格中岛屿的数量。

岛屿总是被水包围,并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。

此外,你可以假设该网格的四条边均被水包围。

  •  对应函数签名如下:

  • 思路:
  • 遍历整个矩阵,每次找到「⼀块陆地」的时候:
  •  说明找到「⼀个岛屿」,记录到最终结果 res⾥⾯
  • 并且将这个陆地相连的所有陆地,也就是这块「岛屿」,全部「变成海洋」。这样的话,我们下次 遍历到这块岛屿的时候,它「已经是海洋」了,不会影响最终结果。
  •  其中「变成海洋」的操作,可以利⽤「深搜」来解决

代码详解:

 解法一:与上面一样,两种解法(类似):

class Solution {int res = 0;public int numIslands(char[][] grid) {int m = grid.length,n = grid[0].length;for(int i = 0;i < m;i++){for(int j = 0;j < n;j++){if(grid[i][j] == '1'){//每次找到一个岛屿记录一下,再将这个岛屿淹没res++;dfs(grid,i,j);}}}return res;}public void dfs(char[][] grid,int i,int j){int m = grid.length,n = grid[0].length;//处理边界情况if(i < 0 || j < 0 || i >= m || j >= n){return ;}if(grid[i][j] == '0') return ;grid[i][j] = '0'; //上下左右去淹没这个岛屿dfs(grid,i - 1,j);dfs(grid,i + 1,j);dfs(grid,i,j - 1);dfs(grid,i,j + 1);}
}

解法二:

class Solution {int res = 0;int[] dx = {0,0,-1,1};int[] dy = {1,-1,0,0};public int numIslands(char[][] grid) {int m = grid.length,n = grid[0].length;for(int i = 0;i < m;i++){for(int j = 0;j < n;j++){if(grid[i][j] == '1'){//说明找到「⼀个岛屿」,记录到最终结果 res⾥⾯res++;dfs(grid,i,j);//将这个岛屿淹没}}}return res;}public void dfs(char[][] grid,int i,int j){int m = grid.length,n = grid[0].length;grid[i][j] = '0'; for(int k = 0;k < 4;k++){int x = i + dx[k],y = j + dy[k];if(x >= 0 && x < m && y >= 0 && y < n&& grid[x][y] != '0'){dfs(grid,x,y);}}}
}

运行结果:

 

三.岛屿的最大面积:

  • 题目链接:695. 岛屿的最大面积 - 力扣(LeetCode)
  • 题目描述:

    给你一个大小为 m x n 的二进制矩阵 grid 。

    岛屿 是由一些相邻的 1 (代表土地) 构成的组合,这里的「相邻」要求两个 1 必须在 水平或者竖直的四个方向上 相邻。你可以假设 grid 的四个边缘都被 0(代表水)包围着。

    岛屿的面积是岛上值为 1 的单元格的数目。

    计算并返回 grid 中最大的岛屿面积。如果没有岛屿,则返回面积为 0 

  • 对应函数签名:

算法思路:

• 遍历整个矩阵,每当遇到⼀块⼟地的时候,就⽤「深搜」或者「宽搜」将与这块⼟地相连的「整个 岛屿」的⾯积计算出来

• 然后在搜索得到的「所有的岛屿⾯积」求⼀个「最⼤值」即可

• 在搜索过程中,为了「防⽌搜到重复的⼟地」:

◦ 可以开⼀个同等规模的「布尔数组」,标记⼀下这个位置是否已经被访问过;

◦ 也可以将原始矩阵的 1 修改成 0 ,但是这样操作会修改原始矩阵。 

代码详解:

 解法一:

class Solution {int maxArea = 0;int count;boolean[][] used;public int maxAreaOfIsland(int[][] grid) {int m = grid.length,n = grid[0].length;used = new boolean[m][n];for(int i = 0;i < m;i++){for(int j = 0;j < n;j++){if(grid[i][j] == 1){//每次找到一个岛屿都要重置计数count = 0;dfs(grid,i,j);maxArea = Math.max(maxArea,count);}}}return maxArea;}public void dfs(int[][] grid,int i,int j){int m = grid.length,n = grid[0].length;//处理边界情况if(i < 0 || j < 0 || i >= m || j >= n){return ;}if(grid[i][j] == 0) return ;if(used[i][j]) return ;used[i][j] = true;count++;dfs(grid,i - 1,j);dfs(grid,i + 1,j);dfs(grid,i,j - 1);dfs(grid,i,j + 1);}
}

解法二:

class Solution {int maxArea = 0;int count = 0;int[] dx = {0,0,-1,1};int[] dy = {1,-1,0,0};boolean[][] used;public int maxAreaOfIsland(int[][] grid) {int m = grid.length,n = grid[0].length;used = new boolean[m][n];for(int i = 0;i < m;i++){for(int j = 0;j < n;j++){if(grid[i][j] == 1){//每次找到一个岛屿都要重置计数count = 0;dfs(grid,i,j);maxArea = Math.max(maxArea,count);}}}return maxArea;}public void dfs(int[][] grid,int i,int j){int m = grid.length,n = grid[0].length;used[i][j] = true;count++;for(int k = 0;k < 4;k++){int x = i + dx[k],y = j + dy[k];//处理不满足条件的情况if(x >= 0 && x < m && y >= 0 && y < n && !used[x][y] && grid[x][y] != 0){dfs(grid,x,y);}}}
}

运行结果:

四.被围绕的区域:

  • 题目链接:130. 被围绕的区域 - 力扣(LeetCode)
  • 题目描述:给你一个 m x n 的矩阵 board ,由若干字符 'X' 和 'O' ,找到所有被 'X' 围绕的区域,并将这些区域里所有的 'O' 用 'X' 填充。

  • 对应函数签名:

  • 算法思路: 
  • 正难则反。 可以先利⽤ dfs 将与边缘相连的 '0' 区域做上标记,然后重新遍历矩阵,将没有标记过的 '0' 修改成 'X' 即可。

 

代码详解:

class Solution {boolean[][] used;public void solve(char[][] board) {int m = board.length,n = board[0].length;used = new boolean[m][n];//分别对应上下左右,标记外围的'O'for(int i = 0;i < n;i++){dfs2(board,0,i);dfs2(board,m - 1,i);}for(int j = 0;j < m;j++){dfs2(board,j,0);dfs2(board,j,n - 1);}for(int i = 0;i < m;i++){for(int j = 0;j < n;j++){if(board[i][j] != 'X' && !used[i][j]){dfs(board,i,j);}}}}//将内部的'O'全部标记为'X'public void dfs(char[][] board,int i,int j){int m = board.length,n = board[0].length;if(i < 0 || j < 0 || i >= m || j >= n){return ;}if(board[i][j] == 'X') return;if(used[i][j]) return ;used[i][j] = true;board[i][j] = 'X';dfs(board,i - 1,j);dfs(board,i + 1,j);dfs(board,i,j - 1);dfs(board,i,j + 1);}//将外围的位置标记为true,后续不会对其进行操作public void dfs2(char[][] board,int i,int j){int m = board.length,n = board[0].length;if(i < 0 || j < 0 || i >= m || j >= n){return ;}if(board[i][j] == 'X') return;if(used[i][j]) return ;used[i][j] = true;dfs2(board,i - 1,j);dfs2(board,i + 1,j);dfs2(board,i,j - 1);dfs2(board,i,j + 1);}
}

运行结果:

五.太平洋大西洋水流问题:

  • 题目链接:417. 太平洋大西洋水流问题 - 力扣(LeetCode)
  • 题目描述:

有一个 m × n 的矩形岛屿,与 太平洋 和 大西洋 相邻。 “太平洋” 处于大陆的左边界和上边界,而 “大西洋” 处于大陆的右边界和下边界。

这个岛被分割成一个由若干方形单元格组成的网格。给定一个 m x n 的整数矩阵 heights , heights[r][c] 表示坐标 (r, c) 上单元格 高于海平面的高度 。

岛上雨水较多,如果相邻单元格的高度 小于或等于 当前单元格的高度,雨水可以直接向北、南、东、西流向相邻单元格。水可以从海洋附近的任何单元格流入海洋。

返回网格坐标 result 的 2D 列表 ,其中 result[i] = [ri, ci] 表示雨水从单元格 (ri, ci) 流动 既可流向太平洋也可流向大西洋 。

  • 对应函数标签: 

  •  算法思路:

正难则反。 如果直接去判断某⼀个位置是否既能到⼤西洋也能到太平洋,会重复遍历很多路径。 我们反着来,从⼤西洋沿岸开始反向 dfs ,这样就能找出那些点可以流向⼤西洋;同理,从太平洋沿 岸也反向 dfs ,这样就能找出那些点可以流向太平洋。那么,被标记两次的点,就是我们要找的结果

 

代码详解:

class Solution {int m ,n;int[] dx = {0,0,1,-1};int[] dy = {1,-1,0,0};public List<List<Integer>> pacificAtlantic(int[][] heights) {m = heights.length;n = heights[0].length;boolean[][] pac = new boolean[m][n];boolean[][] atl = new boolean[m][n];//先搞太平洋for(int j = 0;j < n;j++) dfs(heights,0,j,pac);for(int i = 0;i < m;i++) dfs(heights,i,0,pac);//在搞大西洋for(int i = 0;i < m;i++) dfs(heights,i,n - 1,atl);for(int j  = 0;j < n;j++) dfs(heights,m - 1,j,atl);//再提取结果:List<List<Integer>> res = new ArrayList<>();for(int i = 0;i < m;i++){for(int j = 0;j < n;j++){if(pac[i][j] && atl[i][j]){List<Integer> temp = new ArrayList<>();temp.add(i);temp.add(j);res.add(temp);}}}return res;}public void dfs(int[][] heights,int i,int j,boolean[][] used){used[i][j] = true;for(int k = 0;k < 4;k++){int x = i + dx[k],y = j + dy[k];if(x >= 0 && x < m && y >= 0 && y < n && !used[x][y] && heights[x][y] >= heights[i][j]){dfs(heights,x,y,used);}}}
}

运行结果:

 

结语: 写博客不仅仅是为了分享学习经历,同时这也有利于我巩固知识点,总结该知识点,由于作者水平有限,对文章有任何问题的还请指出,接受大家的批评,让我改进。同时也希望读者们不吝啬你们的点赞+收藏+关注,你们的鼓励是我创作的最大动力!

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

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

相关文章

大语言模型从Scaling Laws到MoE

1、摩尔定律和伸缩法则 摩尔定律&#xff08;Moores law&#xff09;是由英特尔&#xff08;Intel&#xff09;创始人之一戈登摩尔提出的。其内容为&#xff1a;集成电路上可容纳的晶体管数目&#xff0c;约每隔两年便会增加一倍&#xff1b;而经常被引用的“18个月”&#xf…

C++学习第二十二课:STL映射类的深入解析

C学习第二十二课&#xff1a;STL映射类的深入解析 在C标准模板库&#xff08;STL&#xff09;中&#xff0c;映射类&#xff08;std::map和std::multimap&#xff09;是用来存储关联数据的容器。与集合类不同&#xff0c;映射类中的每个元素都是一个键值对&#xff08;key-val…

关于继承~

继承 动物有猫、狗&#xff0c; 猫又分为加菲猫、布偶猫......&#xff1b;狗又有哈士奇、德国牧羊犬...... 我们发现&#xff0c;下一类除了拥有上一类的共性之外&#xff0c;还拥有自己的特性。 于是我们可以利用继承的方式来减少重复的代码 继承的基本语法 class A:p…

buuctf re findKey

参考&#xff1a;http://t.csdnimg.cn/hUKRJ 参考&#xff1a;http://t.csdnimg.cn/kIk4i 32位&#xff0c;ida打开 f5&#xff0c;先不管呢 winmain函数 看看字符串 进到flag那里&#xff0c;没法反编译 寻找 看汇编&#xff0c;发现两个一样的push&#xff08;不理解怎…

【力扣】203、环形链表 II

142. 环形链表 II 要解决这道题&#xff0c;首先需要对问题进行拆解&#xff1a; 确定链表是否存在环确定环的入口点 如何判断是否存在环呢&#xff1f;这个比较容易想到&#xff0c;使用快慢指针即可判断链表是否存在环。我们定义两个指针&#xff1a; ListNode slow head…

CUDA和显卡驱动

1.安装显卡驱动 https://www.nvidia.com/download/index.aspx?langen-us 由于我的显卡是RTX4060&#xff0c;因此先选择RTX40系列&#xff0c;然后选择RTX4060&#xff0c;进行安装 2.查看显卡对应的CUDA CUDA安装地址&#xff1a;https://developer.nvidia.com/cuda-toolk…

buuctf-misc-27.面具下的flag

27.面具下的flag 题目&#xff1a;binwalk分离后&#xff0c;解压vmdk文件,对其中的字符进行翻译 将其放到kali中进行binwalk,可以看到有有隐藏的压缩包文件&#xff0c;我们提取一下 文件放到了主目录下&#xff0c;我们使用对应命令发现有zip文件&#xff0c;然后再使用对应…

前端工程化04-VsCode插件设置总结(持续更)

1、输出语句log设置 log输出、平常你输出log,还必须得打一个console然后再.log()非常不方便&#xff0c;当然我们可以直接输入一个log,但是提示有两个&#xff0c;我们还得上下选择 所以我们直接采用插件的提示 一个clg就可以了 2、括号包裹提示 找到VsCode的settings.js文…

【JavaEE】多线程安全问题

文章目录 1、什么是多线程安全问题2、出现线程不安全的原因2.1 线程在系统中是随机调度&#xff0c;抢占式执行的2.2 多个线程同时修改同一个变量2.3 线程针对变量的修改操作&#xff0c;不是“原子”的2.4 内存可见性问题2.5 指令重排序 3 、如何解决线程安全问题3.1 锁操作3.…

《十三》QT绘图原理双缓冲机制

一、原理与设计 所谓双缓冲机制&#xff0c;是指在绘制控件时&#xff0c;首先将要绘制的内容绘制在一个图片中&#xff0c;再将图片一次性地绘制到控件上。在早期的 Qt 版本中&#xff0c;若直接在控件上进行绘制工作&#xff0c;则在控件重绘时会产生闪烁地现象&#xff0c;控…

C语言之数据结构之栈和队列的运用

目录 1. 用队列实现栈1.1 思路讲解1.2 代码实现 2. 用栈实现队列1.1 思路讲解1.2 代码实现 总结 •͈ᴗ•͈ 个人主页&#xff1a;御翮 •͈ᴗ•͈ 个人专栏&#xff1a;C语言数据结构 •͈ᴗ•͈ 欢迎大家关注和订阅!!! 1. 用队列实现栈 题目描述&#xff1a; 请你仅使用两个…

练习题(2024/5/3)

1对称二叉树 给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 示例 1&#xff1a; 输入&#xff1a;root [1,2,2,3,4,4,3] 输出&#xff1a;true示例 2&#xff1a; 输入&#xff1a;root [1,2,2,null,3,null,3] 输出&#xff1a;false提示&#xff1a; 树中…

鸿蒙通用组件Image简介

鸿蒙通用组件Image简介 图片----Image图片支持三种引用方式设置图片宽高设置图片缩放模式设置图片占位图设置图片重复样式设置图片插值效果 图片----Image Image主要用于在应用中展示图片 Image($r(app.media.app_icon)).width(150) // 设置宽.height(150) // 设置高.objectF…

一文看懂卷积神经网络CNN(1)—前馈神经网络

目录 参考资料 一、神经网络 1、人脑神经网络 2、人工神经网络 3、神经网络的发展历史 二、前馈神经网络 1、神经元 &#xff08;1&#xff09;Sigmoid型函数 ① Logistic函数 ②Tanh函数 ③两个函数形状对比 &#xff08;2&#xff09;ReLU函数 ① 带泄露的ReLU函…

leetcode刷题(3): 动态规划

文章目录 42. 接雨水解题思路c 实现 64. 最小路径和解题思路c 实现 62 不同路径解题思路c 实现 42. 接雨水 题目&#xff1a; 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 示例: 解题思路 使用动态…

大功率双向直流电源的输出电压和电流

双向直流电源&#xff08;Bidirectional DC Power Supply&#xff09;是一种具有双向输出电能的直流电源。与传统的直流电源相比&#xff0c;双向直流电源在输出电能的同时&#xff0c;还具备一定的电流输入能力&#xff0c;从而使其应用范围更加广泛。大功率双向直流电源作为电…

言语 目录

List item List item List item

【业务场景】京东实际场景,频繁GC引起的CPU飙高问题的解决

目录 1.业务介绍 2.判断任务类型 3.CPU飙高的原因 1.业务介绍 本文的业务场景是京东零售线公开的一篇文章&#xff0c;文章内容详细介绍了京东零售线如何将广告相关的定时任务从半小时优化到秒级的&#xff0c;原文链接&#xff1a; 半小时到秒级&#xff0c;京东零售定时…

USP技术提升大语言模型的零样本学习能力

大语言模型&#xff08;LLMs&#xff09;在零样本和少样本学习能力上取得了显著进展&#xff0c;这通常通过上下文学习&#xff08;in-context learning, ICL&#xff09;和提示&#xff08;prompting&#xff09;来实现。然而&#xff0c;零样本性能通常较弱&#xff0c;因为缺…

数据库(MySQL)—— 事务

数据库&#xff08;MySQL&#xff09;—— 事务 什么是事务事务操作未控制事务测试异常情况 控制事务一查看/设置事务提交方式&#xff1a;提交事务回滚事务 控制事务二开启事务提交事务回滚事务 并发事务问题脏读&#xff08;Dirty Read&#xff09;不可重复读&#xff08;Non…