[LeetCode] Surrounded Regions 包围区域
Given a 2D board containing 'X'
and 'O'
(the letter O), capture all regions surrounded by 'X'
.
A region is captured by flipping all 'O'
s into 'X'
s in that surrounded region.
Example:
- X X X X
- X O O X
- X X O X
- X O X X
After running your function, the board should be:
- X X X X
- X X X X
- X X X X
- X O X X
Explanation:
Surrounded regions shouldn’t be on the border, which means that any 'O'
on the border of the board are not flipped to 'X'
. Any 'O'
that is not on the border and it is not connected to an 'O'
on the border will be flipped to 'X'
. Two cells are connected if they are adjacent cells connected horizontally or vertically.
这是道关于 XXOO 的题,有点像围棋,将包住的O都变成X,但不同的是边缘的O不算被包围,跟之前那道 Number of Islands 很类似,都可以用 DFS 来解。刚开始我的思路是 DFS 遍历中间的O,如果没有到达边缘,都变成X,如果到达了边缘,将之前变成X的再变回来。但是这样做非常的不方便,在网上看到大家普遍的做法是扫矩阵的四条边,如果有O,则用 DFS 遍历,将所有连着的O都变成另一个字符,比如 \$,这样剩下的O都是被包围的,然后将这些O变成X,把$变回O就行了。代码如下:
解法一:
- class Solution {
- public:
- void solve(vector<vector<char> >& board) {
- for (int i = ; i < board.size(); ++i) {
- for (int j = ; j < board[i].size(); ++j) {
- if ((i == || i == board.size() - || j == || j == board[i].size() - ) && board[i][j] == 'O')
- solveDFS(board, i, j);
- }
- }
- for (int i = ; i < board.size(); ++i) {
- for (int j = ; j < board[i].size(); ++j) {
- if (board[i][j] == 'O') board[i][j] = 'X';
- if (board[i][j] == '$') board[i][j] = 'O';
- }
- }
- }
- void solveDFS(vector<vector<char> > &board, int i, int j) {
- if (board[i][j] == 'O') {
- board[i][j] = '$';
- if (i > && board[i - ][j] == 'O')
- solveDFS(board, i - , j);
- if (j < board[i].size() - && board[i][j + ] == 'O')
- solveDFS(board, i, j + );
- if (i < board.size() - && board[i + ][j] == 'O')
- solveDFS(board, i + , j);
- if (j > && board[i][j - ] == 'O')
- solveDFS(board, i, j - );
- }
- }
- };
很久以前,上面的代码中最后一个 if 中必须是 j > 1 而不是 j > 0,为啥 j > 0 无法通过 OJ 的最后一个大数据集合,博主开始也不知道其中奥秘,直到被另一个网友提醒在本地机子上可以通过最后一个大数据集合,于是博主也写了一个程序来验证,请参见验证 LeetCode Surrounded Regions 包围区域的DFS方法,发现 j > 0 是正确的,可以得到相同的结果。神奇的是,现在用 j > 0 也可以通过 OJ 了。
下面这种解法还是 DFS 解法,只是递归函数的写法稍有不同,但是本质上并没有太大的区别,参见代码如下:
解法二:
- class Solution {
- public:
- void solve(vector<vector<char>>& board) {
- if (board.empty() || board[].empty()) return;
- int m = board.size(), n = board[].size();
- for (int i = ; i < m; ++i) {
- for (int j = ; j < n; ++j) {
- if (i == || i == m - || j == || j == n - ) {
- if (board[i][j] == 'O') dfs(board, i , j);
- }
- }
- }
- for (int i = ; i < m; ++i) {
- for (int j = ; j < n; ++j) {
- if (board[i][j] == 'O') board[i][j] = 'X';
- if (board[i][j] == '$') board[i][j] = 'O';
- }
- }
- }
- void dfs(vector<vector<char>> &board, int x, int y) {
- int m = board.size(), n = board[].size();
- vector<vector<int>> dir{{,-},{-,},{,},{,}};
- board[x][y] = '$';
- for (int i = ; i < dir.size(); ++i) {
- int dx = x + dir[i][], dy = y + dir[i][];
- if (dx >= && dx < m && dy > && dy < n && board[dx][dy] == 'O') {
- dfs(board, dx, dy);
- }
- }
- }
- };
我们也可以使用迭代的解法,但是整体的思路还是一样的,在找到边界上的O后,然后利用队列 queue 进行 BFS 查找和其相连的所有O,然后都标记上美元号。最后的处理还是先把所有的O变成X,然后再把美元号变回O即可,参见代码如下:
解法三:
- class Solution {
- public:
- void solve(vector<vector<char>>& board) {
- if (board.empty() || board[].empty()) return;
- int m = board.size(), n = board[].size();
- for (int i = ; i < m; ++i) {
- for (int j = ; j < n; ++j) {
- if (i != && i != m - && j != && j != n - ) continue;
- if (board[i][j] != 'O') continue;
- board[i][j] = '$';
- queue<int> q{{i * n + j}};
- while (!q.empty()) {
- int t = q.front(), x = t / n, y = t % n; q.pop();
- if (x >= && board[x - ][y] == 'O') {board[x - ][y] = '$'; q.push(t - n);}
- if (x < m - && board[x + ][y] == 'O') {board[x + ][y] = '$'; q.push(t + n);}
- if (y >= && board[x][y - ] == 'O') {board[x][y - ] = '$'; q.push(t - );}
- if (y < n - && board[x][y + ] == 'O') {board[x][y + ] = '$'; q.push(t + );}
- }
- }
- }
- for (int i = ; i < m; ++i) {
- for (int j = ; j < n; ++j) {
- if (board[i][j] == 'O') board[i][j] = 'X';
- if (board[i][j] == '$') board[i][j] = 'O';
- }
- }
- }
- };
Github 同步地址:
https://github.com/grandyang/leetcode/issues/130
类似题目:
参考资料:
https://leetcode.com/problems/surrounded-regions/
https://leetcode.com/problems/surrounded-regions/discuss/41895/Share-my-clean-Java-Code
LeetCode All in One 题目讲解汇总(持续更新中...)
[LeetCode] Surrounded Regions 包围区域的更多相关文章
- 验证LeetCode Surrounded Regions 包围区域的DFS方法
在LeetCode中的Surrounded Regions 包围区域这道题中,我们发现用DFS方法中的最后一个条件必须是j > 1,如下面的红色字体所示,如果写成j > 0的话无法通过OJ ...
- [LeetCode] 130. Surrounded Regions 包围区域
Given a 2D board containing 'X' and 'O'(the letter O), capture all regions surrounded by 'X'. A regi ...
- [LintCode] Surrounded Regions 包围区域
Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is captured ...
- Surrounded Regions 包围区域——dfs
Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is captured ...
- Leetcode: Surrounded regions
Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is captured ...
- LeetCode: Surrounded Regions 解题报告
Surrounded Regions Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A ...
- LeetCode: Surrounded Regions [130]
[题目] Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is cap ...
- LEETCODE —— Surrounded Regions
Total Accepted: 43584 Total Submissions: 284350 Difficulty: Medium Given a 2D board containing 'X' a ...
- [leetcode]Surrounded Regions @ Python
原题地址:https://oj.leetcode.com/problems/surrounded-regions/ 题意: Given a 2D board containing 'X' and 'O ...
随机推荐
- 从xfire谈WebService接口化编程
前段时间有博友在看我的博文<WebService入门案例>后,发邮件问我关于WebService 接口在java中的开发,以及在实际生产环境中的应用.想想自己入职也有一段时间了,似乎也该总 ...
- 记录软件工程课程项目开发时遇到的各种小问题(django)
1.python manage.py makemigrations 无效/无法检测出model的变化 在修改了models.py之后,我们想要更新数据库的表,使用了python manage.py m ...
- .NET Core 1.1 发布 文档下载资源汇总
.NET Core 1.1 RTM 版2016/11/16 发布.对应发布 ASP.NET Core 1.1 .EF Core 1.1. 你可以通过Visual Studio 2015, Visual ...
- ASP.NET的视图(Razor)循环产生html代码
需要要视图中Razor语法,循环产生一些html代码. 产生后的html是这样的: <li data-transition="> <img src="~/Cont ...
- C# 当前系统的多管理账户测判断
using (DirectoryEntry comp = new DirectoryEntry("WinNT://" + Environment.MachineName + &qu ...
- shell 带签名请求,yii 处理带签名的请求
处理请求 class TestController extends Controller { public function init() { if(!YII_ENV_DEV){ throw new ...
- 在MongoDB的MapReduce上踩过的坑
太久没动这里,目前人生处于一个新的开始.这次博客的内容很久前就想更新上来,但是一直没找到合适的时间点(哈哈,其实就是懒),主要内容集中在使用Mongodb时的一些隐蔽的MapReduce问题: 1.R ...
- 《连载 | 物联网框架ServerSuperIO教程》2.服务实例的配置参数说明
1.C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍 <连载 | 物联网框架ServerSuperIO教程>1.4种通讯模式机制 一.综述 SuperIO(SIO)定位 ...
- iOS 开发者账号到期续费流程
1.登录developer.apple.com,查看到期时间 2.到期提醒通知,点击Renew Membership续费(一般提前一个月提醒续费) 3.个人开发者账号续费需要支付 688人民币/年(9 ...
- iOS角度与弧度转换
在iOS中图片的旋转单位为弧度而不是角度,所以经常会在两者之间进行转换 弧度转角度 #define RADIANS_TO_DEGREES(radians) ((radians) * (180.0 / ...