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:

  1. X X X X
  2. X O O X
  3. X X O X
  4. X O X X

After running your function, the board should be:

  1. X X X X
  2. X X X X
  3. X X X X
  4. 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就行了。代码如下:

解法一:

  1. class Solution {
  2. public:
  3. void solve(vector<vector<char> >& board) {
  4. for (int i = ; i < board.size(); ++i) {
  5. for (int j = ; j < board[i].size(); ++j) {
  6. if ((i == || i == board.size() - || j == || j == board[i].size() - ) && board[i][j] == 'O')
  7. solveDFS(board, i, j);
  8. }
  9. }
  10. for (int i = ; i < board.size(); ++i) {
  11. for (int j = ; j < board[i].size(); ++j) {
  12. if (board[i][j] == 'O') board[i][j] = 'X';
  13. if (board[i][j] == '$') board[i][j] = 'O';
  14. }
  15. }
  16. }
  17. void solveDFS(vector<vector<char> > &board, int i, int j) {
  18. if (board[i][j] == 'O') {
  19. board[i][j] = '$';
  20. if (i > && board[i - ][j] == 'O')
  21. solveDFS(board, i - , j);
  22. if (j < board[i].size() - && board[i][j + ] == 'O')
  23. solveDFS(board, i, j + );
  24. if (i < board.size() - && board[i + ][j] == 'O')
  25. solveDFS(board, i + , j);
  26. if (j > && board[i][j - ] == 'O')
  27. solveDFS(board, i, j - );
  28. }
  29. }
  30. };

很久以前,上面的代码中最后一个 if 中必须是 j > 1 而不是 j > 0,为啥 j > 0 无法通过 OJ 的最后一个大数据集合,博主开始也不知道其中奥秘,直到被另一个网友提醒在本地机子上可以通过最后一个大数据集合,于是博主也写了一个程序来验证,请参见验证 LeetCode Surrounded Regions 包围区域的DFS方法,发现 j > 0 是正确的,可以得到相同的结果。神奇的是,现在用 j > 0  也可以通过 OJ 了。

下面这种解法还是 DFS 解法,只是递归函数的写法稍有不同,但是本质上并没有太大的区别,参见代码如下:

解法二:

  1. class Solution {
  2. public:
  3. void solve(vector<vector<char>>& board) {
  4. if (board.empty() || board[].empty()) return;
  5. int m = board.size(), n = board[].size();
  6. for (int i = ; i < m; ++i) {
  7. for (int j = ; j < n; ++j) {
  8. if (i == || i == m - || j == || j == n - ) {
  9. if (board[i][j] == 'O') dfs(board, i , j);
  10. }
  11. }
  12. }
  13. for (int i = ; i < m; ++i) {
  14. for (int j = ; j < n; ++j) {
  15. if (board[i][j] == 'O') board[i][j] = 'X';
  16. if (board[i][j] == '$') board[i][j] = 'O';
  17. }
  18. }
  19. }
  20. void dfs(vector<vector<char>> &board, int x, int y) {
  21. int m = board.size(), n = board[].size();
  22. vector<vector<int>> dir{{,-},{-,},{,},{,}};
  23. board[x][y] = '$';
  24. for (int i = ; i < dir.size(); ++i) {
  25. int dx = x + dir[i][], dy = y + dir[i][];
  26. if (dx >= && dx < m && dy > && dy < n && board[dx][dy] == 'O') {
  27. dfs(board, dx, dy);
  28. }
  29. }
  30. }
  31. };

我们也可以使用迭代的解法,但是整体的思路还是一样的,在找到边界上的O后,然后利用队列 queue 进行 BFS 查找和其相连的所有O,然后都标记上美元号。最后的处理还是先把所有的O变成X,然后再把美元号变回O即可,参见代码如下:

解法三:

  1. class Solution {
  2. public:
  3. void solve(vector<vector<char>>& board) {
  4. if (board.empty() || board[].empty()) return;
  5. int m = board.size(), n = board[].size();
  6. for (int i = ; i < m; ++i) {
  7. for (int j = ; j < n; ++j) {
  8. if (i != && i != m - && j != && j != n - ) continue;
  9. if (board[i][j] != 'O') continue;
  10. board[i][j] = '$';
  11. queue<int> q{{i * n + j}};
  12. while (!q.empty()) {
  13. int t = q.front(), x = t / n, y = t % n; q.pop();
  14. if (x >= && board[x - ][y] == 'O') {board[x - ][y] = '$'; q.push(t - n);}
  15. if (x < m - && board[x + ][y] == 'O') {board[x + ][y] = '$'; q.push(t + n);}
  16. if (y >= && board[x][y - ] == 'O') {board[x][y - ] = '$'; q.push(t - );}
  17. if (y < n - && board[x][y + ] == 'O') {board[x][y + ] = '$'; q.push(t + );}
  18. }
  19. }
  20. }
  21. for (int i = ; i < m; ++i) {
  22. for (int j = ; j < n; ++j) {
  23. if (board[i][j] == 'O') board[i][j] = 'X';
  24. if (board[i][j] == '$') board[i][j] = 'O';
  25. }
  26. }
  27. }
  28. };

Github 同步地址:

https://github.com/grandyang/leetcode/issues/130

类似题目:

Number of Islands

Walls and Gates

参考资料:

https://leetcode.com/problems/surrounded-regions/

https://leetcode.com/problems/surrounded-regions/discuss/41895/Share-my-clean-Java-Code

https://leetcode.com/problems/surrounded-regions/discuss/41825/Simple-BFS-solution-easy-to-understand

https://leetcode.com/problems/surrounded-regions/discuss/41612/A-really-simple-and-readable-C%2B%2B-solutionuff0conly-cost-12ms

LeetCode All in One 题目讲解汇总(持续更新中...)

[LeetCode] Surrounded Regions 包围区域的更多相关文章

  1. 验证LeetCode Surrounded Regions 包围区域的DFS方法

    在LeetCode中的Surrounded Regions 包围区域这道题中,我们发现用DFS方法中的最后一个条件必须是j > 1,如下面的红色字体所示,如果写成j > 0的话无法通过OJ ...

  2. [LeetCode] 130. Surrounded Regions 包围区域

    Given a 2D board containing 'X' and 'O'(the letter O), capture all regions surrounded by 'X'. A regi ...

  3. [LintCode] Surrounded Regions 包围区域

    Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is captured ...

  4. Surrounded Regions 包围区域——dfs

    Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is captured ...

  5. Leetcode: Surrounded regions

    Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is captured ...

  6. LeetCode: Surrounded Regions 解题报告

    Surrounded Regions Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A ...

  7. LeetCode: Surrounded Regions [130]

    [题目] Given a 2D board containing 'X' and 'O', capture all regions surrounded by 'X'. A region is cap ...

  8. LEETCODE —— Surrounded Regions

    Total Accepted: 43584 Total Submissions: 284350 Difficulty: Medium Given a 2D board containing 'X' a ...

  9. [leetcode]Surrounded Regions @ Python

    原题地址:https://oj.leetcode.com/problems/surrounded-regions/ 题意: Given a 2D board containing 'X' and 'O ...

随机推荐

  1. 从xfire谈WebService接口化编程

    前段时间有博友在看我的博文<WebService入门案例>后,发邮件问我关于WebService 接口在java中的开发,以及在实际生产环境中的应用.想想自己入职也有一段时间了,似乎也该总 ...

  2. 记录软件工程课程项目开发时遇到的各种小问题(django)

    1.python manage.py makemigrations 无效/无法检测出model的变化 在修改了models.py之后,我们想要更新数据库的表,使用了python manage.py m ...

  3. .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 ...

  4. ASP.NET的视图(Razor)循环产生html代码

    需要要视图中Razor语法,循环产生一些html代码. 产生后的html是这样的: <li data-transition="> <img src="~/Cont ...

  5. C# 当前系统的多管理账户测判断

    using (DirectoryEntry comp = new DirectoryEntry("WinNT://" + Environment.MachineName + &qu ...

  6. shell 带签名请求,yii 处理带签名的请求

    处理请求 class TestController extends Controller { public function init() { if(!YII_ENV_DEV){ throw new ...

  7. 在MongoDB的MapReduce上踩过的坑

    太久没动这里,目前人生处于一个新的开始.这次博客的内容很久前就想更新上来,但是一直没找到合适的时间点(哈哈,其实就是懒),主要内容集中在使用Mongodb时的一些隐蔽的MapReduce问题: 1.R ...

  8. 《连载 | 物联网框架ServerSuperIO教程》2.服务实例的配置参数说明

    1.C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍  <连载 | 物联网框架ServerSuperIO教程>1.4种通讯模式机制 一.综述 SuperIO(SIO)定位 ...

  9. iOS 开发者账号到期续费流程

    1.登录developer.apple.com,查看到期时间 2.到期提醒通知,点击Renew Membership续费(一般提前一个月提醒续费) 3.个人开发者账号续费需要支付 688人民币/年(9 ...

  10. iOS角度与弧度转换

    在iOS中图片的旋转单位为弧度而不是角度,所以经常会在两者之间进行转换 弧度转角度 #define RADIANS_TO_DEGREES(radians) ((radians) * (180.0 / ...