floodfill算法_dfs

server/2025/1/8 1:30:34/

之前我们用BFS解决floodfil算法

BFS 解决 FloodFill 算法_图像渲染_岛屿数量_被围绕的区域-CSDN博客

下面我们用dfs进行解决

733. 图像渲染

从初始位置开始dfs,找和它值相同的,并在dfs过程中把值改为目标值。因为会改变原数组,不用vis记录经过路径也可以。

边界情况:如果初始位置值结束目标值,那在dfs过程中不会改值,就会出现走重复路径的情况。遇到这种情况,其实是不用进行dfs,原数组就是最终结果。

class Solution {vector<int> dx={1,-1,0,0};vector<int> dy={0,0,1,-1};int m,n;int prev,color;
public:vector<vector<int>> floodFill(vector<vector<int>>& image, int sr, int sc, int _color) {//1.记录要修改的像素值prev=image[sr][sc];color=_color;//2.处理边界情况 (第一个就是要变成的颜色,就没必要处理)if(prev==color) return image;//3.二维数组边界 m行数 n列数m=image.size(),n=image[0].size();dfs(image,sr,sc);return image;}void dfs(vector<vector<int>>& image, int i, int j){//进入dfs就更改值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&&image[x][y]==prev){dfs(image,x,y);}}return;}
};

200. 岛屿数量

class Solution {int dx[4]={1,-1,0,0};int dy[4]={0,0,1,-1};bool vis[301][301];int m,n;
public:int numIslands(vector<vector<char>>& grid) {int re=0;m=grid.size(),n=grid[0].size();for(int i=0;i<m;i++){for(int j=0;j<n;j++){if(grid[i][j]=='1'&&!vis[i][j]){dfs(grid,i,j); re++;}}}return re;}void dfs(vector<vector<char>>& grid,int i,int j){vis[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&&!vis[x][y]&&grid[x][y]=='1'){dfs(grid,x,y);}}return;}
};

695. 岛屿的最大面积

每遍历到一个1且没有标记过,就从该位置进行dfs,进入dfs就标记并记录当前面积,dfs结束获取这一块1的面积。

class Solution {int dx[4]={0,0,1,-1};int dy[4]={1,-1,0,0};int m,n;bool vis[51][51];//记录“1”块的面积int count=0;
public:int maxAreaOfIsland(vector<vector<int>>& grid) {m=grid.size(),n=grid[0].size();int re=0;for(int i=0;i<m;i++){for(int j=0;j<n;j++){if(!vis[i][j]&&grid[i][j]==1){count=0;dfs(grid,i,j);re=max(re,count);}}}return re;}int dfs(vector<vector<int>>& grid,int i,int j){//进入一个面积+1 标记count++;vis[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&&!vis[x][y]&&grid[x][y]==1){dfs(grid,x,y);}}return count;}
};

130. 被围绕的区域

class Solution {int dx[4] = {1, -1, 0, 0};int dy[4] = {0, 0, 1, -1};bool vis[201][201];int m, n;public:void solve(vector<vector<char>>& board) {m = board.size(), n = board[0].size();// 1. 先处理边界上的 'O' 联通块 对应的vis改为1for (int j = 0; j < n; j++) {if (board[0][j] == 'O')dfs(board, 0, j);if (board[m - 1][j] == 'O'&&!vis[m-1][j])dfs(board, m - 1, j);}for (int i = 0; i < m; i++) {if (board[i][0] == 'O'&&!vis[i][0])dfs(board, i, 0);if (board[i][n - 1] == 'O'&&!vis[i][n-1])dfs(board, i, n - 1);}// 2. 更改for (int i = 0; i < m; i++)for (int j = 0; j < n; j++)if (board[i][j] == 'O'&&!vis[i][j])board[i][j] = 'X';}void dfs(vector<vector<char>>& board, int i, int j){vis[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 &&!vis[x][y]&&board[x][y]=='O'){dfs(board,x,y);}}return;}
};

417. 太平洋大西洋水流问题

每个格子上都有水,水从高处流向低处,问那个格子的水可以流向太平洋和大西洋。

思路一:对每个格子都进行dfs,找可以流向两个洋的格子。

时间复杂度太高,不推荐

class Solution {int dx[4]={1,-1,0,0};int dy[4]={0,0,1,-1};vector<vector<int>> re;bool vis[201][201];bool P=false,A=false;int m,n;
public:vector<vector<int>> pacificAtlantic(vector<vector<int>>& h) {m=h.size(),n=h[0].size();for(int i=0;i<m;i++){for(int j=0;j<n;j++){P=false,A=false;vis[i][j]=true;dfs(h,i,j);vis[i][j]=false;if(P&&A) re.push_back({i,j});}}return re;}void dfs(vector<vector<int>>& h,int i,int j){if(i==m-1||j==n-1)A=true;if(i==0||j==0) P=true;if(A&&P) return;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&&!vis[x][y]&&h[x][y]<=h[i][j]){vis[x][y]=true;dfs(h,x,y);vis[x][y]=false;}}return;}
};

方法二:正难则反

水从高到低,反过来水从低到高,路径是一样的。

1.看临边是Pac洋的格子从低到高走,四周的格子比它大就可以走。(用绿色标记走过的格子)

2.看临边是Atl洋的格子从低到高走,四周的格子比它大就可以走。(用红色标记走过的格子)

它们相交的格子就是可以流向两个洋的格子

因为有两种标记,vis图要有两个。对靠近Pac洋的格子dfs进行标记,还要对靠近Atl洋的格子dfs进行标记。两个vis图都为true的位置为目标位置。

注意:vis1 vis2要定义成全局变量 进行传参 保存改变结果

或者在函数内定义成vector<vector<bool> vis(m,vector<bool>(n));进行引用&传参

class Solution {int dx[4] = {1, -1, 0, 0};int dy[4] = {0, 0, 1, -1};vector<vector<int>> re;bool vis1[201][201];bool vis2[201][201];int m, n;public:vector<vector<int>> pacificAtlantic(vector<vector<int>>& h) {m = h.size();n = h[0].size();// // 初始化访问数组// memset(vis1, false, sizeof(vis1));// memset(vis2, false, sizeof(vis2));// 从每个边缘开始 DFSfor (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (i == 0 || j == 0)  // 太平洋dfs(h, i, j, vis1);if (i == m - 1 || j == n - 1)  // 大西洋dfs(h, i, j, vis2);}}// 收集同时到达太平洋和大西洋的坐标for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (vis1[i][j] && vis2[i][j]) re.push_back({i, j});}}return re;}void dfs(vector<vector<int>>& h, int i, int j, bool vis[201][201]) {vis[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 && !vis[x][y] && h[x][y] >=h[i][j]) {dfs(h, x, y, vis);}}}
};

529. 扫雷游戏

在eg1中click位置点击,该格四周没有地雷,改为B,并向四周dfs找E未挖出的格子。如果该格周围有地雷就改为周围地雷数,不进行dfs。

点到地雷就更改并返回。

我们可以先用map数组记录每个格子四周有多少地雷,这样在判断该格四周是否有地雷就不用遍历周围 直接看map中是否有数就行。

class Solution {int map[51][51]={0};int dx[8]={1,-1,0,0,1,-1,1,-1};int dy[8]={0,0,1,-1,-1,1,1,-1};int m,n;
public:vector<vector<char>> updateBoard(vector<vector<char>>& board, vector<int>& click) {m=board.size(),n=board[0].size();//在map图中记录该格周围有多少地雷for(int i=0;i<m;i++)for(int j=0;j<n;j++){if(board[i][j]=='M'){for(int k=0;k<8;k++){int x=i+dx[k],y=j+dy[k];if(x >= 0 && x < m && y >= 0 && y < n ) map[x][y]++;}}}int x=click[0],y=click[1];//遇到地雷更改并返回if(board[x][y]=='M'){board[x][y]='X';return board;}//遇到E分两情况 四周有地雷和无地雷else{dfs(board,x,y);}return board;}void dfs(vector<vector<char>>& board,int i,int j){//1.周围有地雷 E变为周围地雷数if(map[i][j]) board[i][j]='0'+map[i][j];//2.没有地雷 E变B并以该各为起点dfs找E未挖出的格else {board[i][j]='B';for(int k=0;k<8;k++){int x=i+dx[k],y=j+dy[k];if(x >= 0 && x < m && y >= 0 && y < n &&board[x][y]=='E'){dfs(board,x,y);}}}return;}
};

LCR 130. 衣橱整理

从(0,0)进行dfs遍历所有格子,找出符合条件的格子。不能重复记录符合条件的格子,用vis记录走过的路径。

class Solution {int dx[4]={0,0,1,-1};int dy[4]={1,-1,0,0};bool vis[101][101];int m,n,cnt;int re=0;
public:int wardrobeFinishing(int _m, int _n, int _cnt) {m=_m,n=_n,cnt=_cnt;dfs(0,0);return re;}void dfs(int i,int j){vis[i][j]=true;int tmp1=i,tmp2=j,sum=0;while(tmp1||tmp2){sum+=tmp1%10;tmp1/=10;sum+=tmp2%10;tmp2/=10;}if(sum>cnt) return;else re++;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&&!vis[x][y]){dfs(x,y);}}return;}
};


http://www.ppmy.cn/server/156648.html

相关文章

明源地产ERP VisitorWeb_XMLHTTP.aspx Sql注入漏洞复现(附脚本)

0x01 产品描述: ‌明源地产ERP系统‌是一款专为房地产行业设计的企业资源规划系统,旨在打通企业各个部门之间的信息壁垒,实现资源优化配置,提高运营效率。该系统集房源、客源、销售、财务、人力资源等功能于一身,满足房地产企业在多个环节的精细化管理需求‌0x02 漏洞描述…

unity 播放 序列帧图片 动画

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、方法一&#xff1a;代码控制播放序列帧1、设置图片属性2、创建Image组件3、简单的代码控制4、挂载代码并赋值 二、方法二&#xff1a;直接使用1.Image上添加…

JS数组转字符串(3种方法)

JavaScript 允许数组与字符串之间相互转换。其中 Array 方法对象定义了 3 个方法&#xff0c;可以把数组转换为字符串&#xff0c;如表所示。 Array 对象的数组与字符串相互转换方法 数组方法 说明 toString() 将数组转换成一个字符串 toLocalString() 把数组转换成本地约定的…

大麦抢票科技狠活

仅供学习参考&#xff0c;切勿再令您所爱的人耗费高昂的价格去购置黄牛票 ⚠️核心内容参考: 据悉&#xff0c;于购票环节&#xff0c;大麦凭借恶意流量清洗技术&#xff0c;于网络层实时甄别并阻拦凭借自动化手段发起下单请求的流量&#xff0c;强化对刷票脚本、刷票软件以及…

文档解析工具:如何将PDF文档转换为Markdown格式?

将PDF文档转换为Markdown格式可以通过多种工具和方法实现&#xff0c;以下是一些推荐的工具和步骤&#xff1a; 推荐工具&#xff1a;TextIn 特点&#xff1a;支持在线转换&#xff0c;无需下载软件&#xff0c;能够保留PDF的原始格式和内容。使用步骤&#xff1a; 访问Text…

大数据-267 实时数仓 - ODS Lambda架构 Kappa架构 核心思想

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; Java篇开始了&#xff01; MyBatis 更新完毕目前开始更新 Spring&#xff0c;一起深入浅出&#xff01; 目前已经更新到了&#xff1a; H…

spring boot IDEA启动两个端口服务nginx负载

输入-Dserver.port8153 设置新的服务的端口号 在service里面管理多个端口 niginx配置

LabVIEW在反馈控制时如何解决带约束的控制问题

在LabVIEW中&#xff0c;解决带约束的反馈控制问题通常需要使用先进的控制算法或特定的方法来满足约束条件&#xff0c;同时保证控制系统的性能和稳定性。以下是解决这类问题的一些常用方法和步骤&#xff1a; ​ 1. 定义控制问题及约束条件 确定被控对象的动态特性&#xff08…