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

成都网站营销批发在线客服

成都网站营销批发,在线客服,网站建设和维护视频,摄影手机网站模板洪水填充是一种很简单的技巧,设置路径信息进行剪枝和统计,类似感染的过程。路径信息不撤销,来保证每一片的感染过程可以得到区分。看似是暴力递归过程,其实时间复杂度非常好,遍历次数和样本数量的规模一致。 下面通过…

洪水填充是一种很简单的技巧,设置路径信息进行剪枝和统计,类似感染的过程。路径信息不撤销,来保证每一片的感染过程可以得到区分。看似是暴力递归过程,其实时间复杂度非常好,遍历次数和样本数量的规模一致。

下面通过一些题目来加深理解。

题目一

测试链接:https://leetcode.cn/problems/number-of-islands/

分析:洪水填充更可以看作是一个感染过程,将空间中符合条件的位置感染成我们需要的东西。这道题可以从是陆地的地方深度优先开始感染,将陆地感染成用以区分不再是陆地的信号。当遍历完整个空间后,我们就知道有多少个岛屿。代码如下。

class Solution {
public:int arr[5] = {0, 1, 0, -1, 0};void dfs(int i, int j, int row, int column, vector<vector<char>>& grid){if(i < 0 || i >= row || j < 0 || j >= column || grid[i][j] != '1'){return;}grid[i][j] = '0';for(int index = 0;index < 4;++index){dfs(i+arr[index], j+arr[index+1], row, column, grid);}}int numIslands(vector<vector<char>>& grid) {int row = grid.size();int column = grid[0].size();int ans = 0;for(int i = 0;i < row;++i){for(int j = 0;j < column;++j){if(grid[i][j] == '1'){++ans;dfs(i, j, row, column, grid);}}}return ans;}
};

其中,arr数组是用于遍历一个位置的周围即上下左右;在numIslands方法中调用dfs方法时,岛屿数加1,相当于重新感染一个新岛屿。

题目二

测试链接:https://leetcode.cn/problems/surrounded-regions/

分析:这个题可以看出,如果一个O和矩阵的四周存在的O相连,则这个O不会被替换成X,那么我们可以对矩阵的四周,也就是上下左右四个边界的O,使用深度优先进行感染,将其设为一个用以区分的信号。最后将矩阵中所有没有被更新为信号的O,更新为X;将更新为信号的位置更新为O。代码如下。

class Solution {
public:int arr[5] = {0, 1, 0, -1, 0};void dfs(int i, int j, int row, int column, vector<vector<char>>& board){if(i < 0 || i >= row || j < 0 || j >= column || board[i][j] != 'O'){return;}board[i][j] = 'z';for(int index = 0;index < 4;++index){dfs(i+arr[index], j+arr[index+1], row, column, board);}}void solve(vector<vector<char>>& board) {int row = board.size();int column = board[0].size();for(int i = 0;i < column;++i){if(board[0][i] == 'O'){dfs(0, i, row, column, board);}}for(int i = 0;i < column;++i){if(board[row-1][i] == 'O'){dfs(row-1, i, row, column, board);}}for(int i = 0;i < row;++i){if(board[i][0] == 'O'){dfs(i, 0, row, column, board);}}for(int i = 0;i < row;++i){if(board[i][column-1] == 'O'){dfs(i, column-1, row, column, board);}}for(int i = 0;i < row;++i){for(int j = 0;j < column;++j){if(board[i][j] == 'O'){board[i][j] = 'X';}else if(board[i][j] == 'z'){board[i][j] = 'O';}}}}
};

其中,信号为z,也就是说在最后更新时,O更新为X,z更新为O;最开始的4个for循环是对上下左右四个边界的O进行感染。

题目三

测试链接:https://leetcode.cn/problems/making-a-large-island/

分析:这道题涉及到岛屿的面积,所以在感染时,我们设置的信号应该是岛屿的编号。这样方便对不同编号的岛屿进行面积统计。在感染之后,我们得到了岛屿的数量,以及每个岛屿的面积。这时,遍历空间中那些不是岛屿的位置并进行判定。判定过程为对这些位置上下左右是否连接了岛屿,如果连接了,则加上这个岛屿的面积,注意不要重复计算岛屿面积,最后再加上这个位置的面积,也就是再加1。遍历空间中不是岛屿的位置,最终得到最大值。代码如下。

class Solution {
public:int area[125003] = {0};set<int> used;int arr[5] = {0, 1, 0, -1, 0};void dfs(int i, int j, int n, vector<vector<int>>& grid, int number){if(i < 0 || i >= n || j < 0 || j >= n || grid[i][j] != 1){return;}grid[i][j] = number;++area[number];for(int index = 0;index < 4;++index){dfs(i+arr[index], j+arr[index+1], n, grid, number);}}int largestIsland(vector<vector<int>>& grid) {int n = grid.size();int number = 2;int ans = 0;int temp_ans;for(int i = 0;i < n;++i){for(int j = 0;j < n;++j){if(grid[i][j] == 1){dfs(i, j, n, grid, number);++number;}}}for(int i = 2;i < number;++i){ans = ans > area[i] ? ans : area[i];}for(int i = 0;i < n;++i){for(int j = 0;j < n;++j){if(grid[i][j] == 0){temp_ans = 0;for(int index = 0;index < 4;++index){if(i+arr[index] >= 0 && i+arr[index] < n && j+arr[index+1] >= 0 && j+arr[index+1] < n &&grid[i+arr[index]][j+arr[index+1]] != 0 && used.count(grid[i+arr[index]][j+arr[index+1]]) == 0){temp_ans += area[grid[i+arr[index]][j+arr[index+1]]];used.insert(grid[i+arr[index]][j+arr[index+1]]);}}++temp_ans;ans = ans > temp_ans ? ans : temp_ans;used.clear();}}}return ans;}
};

其中,used用来判断一个岛屿是否已经被加过;number是当前岛屿的编号,因为0和1已经被使用所以number从2开始;需要将ans的初始值设为最大岛屿值是因为有可能全部是陆地,也就是没有连接的机会,那么这时的最大值自然就是最大岛屿值。

题目四

测试链接:https://leetcode.cn/problems/bricks-falling-when-hit/

分析:这个题我们看到了稳定和掉落这两个状态,很容易想到使用并查集。使用并查集的思路是,在没有打落砖块的时候,使用并查集将稳定的砖块设为一个集合,不稳定的砖块设为一个集合。最开始顶部的砖块是稳定的,然后相邻砖块之间归为一个集合。设置一个是否稳定的数组,顶部砖块是稳定的设为true,其他设为false。在union方法合并的时候,更新代表元素的是否稳定数组。遍历完砖块后,查询有多少个砖块的集合的代表元素是稳定的,这时候得到没有打入炮弹时的稳定砖块数目。实际上这个数目应该是统计的砖块数目。然后每打落一个砖块,重新统计一次稳定的数目,然后用上一次稳定的数目减此次稳定的数目再减1,因为打落的砖块是消失不计入不稳定的数目,但是这个方法复杂度太高,会超时。所以使用洪水填充加时光倒流。我们需要得到每次炮弹掉落的砖块数目,那么可以在所有炮弹打落之后往回加,也就是从最后一个炮弹开始,每一个炮弹没打之后多了多少稳定的砖块。主要流程就是,首先,将所有炮弹打出去,也就是把炮弹的指向位置的网格值减1,然后将顶部的网格,也就是已经稳定的网格开始洪水填充现存的砖块,将网格值更新为2(什么都行主要要区分1),然后开始时光倒流,从最后一个炮弹往前遍历,把炮弹指向位置处加1,如果为0,则代表之前这个位置没有砖块,continue;如果大于0,则判定这个位置的上下左右是否有等于2的砖块,因为2代表了现在稳定的砖块,如果有,则从炮弹恢复的位置开始洪水填充将1更新为2,更新的数目减1则为这个炮弹对应掉落的砖块数目,减1是因为炮弹指向位置不计入掉落的砖块数目。代码如下。

class Solution {
public:int arr[5] = {0, 1, 0, -1, 0};int dfs(int i, int j, int row, int column, vector<vector<int>>& grid){if(i < 0 || i >= row || j < 0 || j >= column || grid[i][j] != 1){return 0;}int num = 1;grid[i][j] = 2;for(int index = 0;index < 4;++index){num += dfs(i+arr[index], j+arr[index+1], row, column, grid);}return num;}vector<int> hitBricks(vector<vector<int>>& grid, vector<vector<int>>& hits) {vector<int> ans;int row = grid.size();int column = grid[0].size();int length = hits.size();ans.assign(length, 0);for(int i = 0;i < length;++i){--grid[hits[i][0]][hits[i][1]];}for(int i = 0;i < column;++i){if(grid[0][i] == 1){dfs(0, i, row, column, grid);}}for(int i = length-1;i >= 0;--i){++grid[hits[i][0]][hits[i][1]];if(grid[hits[i][0]][hits[i][1]] == 0){continue;}if(hits[i][0] == 0){ans[i] = dfs(hits[i][0], hits[i][1], row, column, grid) - 1;}else if(hits[i][0] >= 0 && hits[i][0] < row && hits[i][1]+1 >= 0 && hits[i][1]+1 < column && grid[hits[i][0]][hits[i][1]+1] == 2){ans[i] = dfs(hits[i][0], hits[i][1], row, column, grid) - 1;}else if(hits[i][0] >= 0 && hits[i][0] < row && hits[i][1]-1 >= 0 && hits[i][1]-1 < column && grid[hits[i][0]][hits[i][1]-1] == 2){ans[i] = dfs(hits[i][0], hits[i][1], row, column, grid) - 1;}else if(hits[i][0]+1 >= 0 && hits[i][0]+1 < row && hits[i][1] >= 0 && hits[i][1] < column && grid[hits[i][0]+1][hits[i][1]] == 2){ans[i] = dfs(hits[i][0], hits[i][1], row, column, grid) - 1;}else if(hits[i][0]-1 >= 0 && hits[i][0]-1 < row && hits[i][1] >= 0 && hits[i][1] < column && grid[hits[i][0]-1][hits[i][1]] == 2){ans[i] = dfs(hits[i][0], hits[i][1], row, column, grid) - 1;}}return ans;}
};

其中,在恢复炮弹指向位置时大于0的判定中,如果指向位置本身就是顶部位置,那么不管这个位置的上下左右是否有为2的位置,直接填充。

http://www.bjxfkj.com.cn/article/101138.html

相关文章:

  • wordpress微信红包百度快照优化公司
  • 用阿里云服务器做盗版小说网站吗百度官方网站登录
  • 福建建设执业注册管理中心网站广告公司招聘
  • 智能营销型网站制作外贸怎么找客户资源
  • 徐州智能模板建站域名ip地址在线查询
  • 有关做粪污处理设备的企业网站精准客源引流平台
  • 淘宝的网站怎么做的好惠州网站seo
  • 天津关键词优化网站怎样提高百度推广排名
  • 网站系统怎么做的杭州互联网公司排名榜
  • 沈阳中天建设集团网站外包公司是正规公司吗
  • 做网站开发使用百分比的好处免费奖励自己的网站
  • 阿里云可以做网站么线上营销策划案例
  • 网站建设推广加盟全球搜是什么公司
  • 情侣打扑克视频免费版appseo是哪个英文的缩写
  • 哈市住房和建设局网站什么软件可以推广自己的产品
  • 上海专业网站建设服务百度推广平台收费标准
  • 5个月的新站网站被k了会怎么样东营优化公司
  • 小程序生成器seo推广任务小结
  • 邯郸网站建设推广专业seo网站优化推广排名教程
  • 网站服务器放置地怎么填写seo成功的案例和分析
  • 商场设计调研西安seo关键词排名
  • 中企动力企业邮箱网址上海免费关键词排名优化
  • 做润滑油网站图片网络营销中的seo与sem
  • 邢台地区网站建设口碑好广州网站建设工作室
  • 黑龙江疫情风险等级windows优化大师有毒吗
  • 无锡网站制作电话如何在百度上发布自己的广告
  • 给个网站你知道企业建站要多少钱
  • 做网站需要哪个系统seo排名哪家公司好
  • 中国版动漫网站模板在线客服系统
  • 做鲜榨果汁店网站中国培训网是国家公认的吗