题目:
给你一个 m x n 的矩阵 board ,由若干字符 'X' 和 'O' 组成,捕获 所有 被围绕的区域:
- 连接:一个单元格与水平或垂直方向上相邻的单元格连接。
- 区域:连接所有
'0'的单元格来形成一个区域。 - 围绕:如果您可以用
'X'单元格 连接这个区域,并且区域中没有任何单元格位于board边缘,则该区域被'X'单元格围绕。
通过将输入矩阵 board 中的所有 'O' 替换为 'X' 来 捕获被围绕的区域。

- 初始化和边界检查:
- 检查
board是否为空或长度为零。- 获取矩阵的行数
m和列数n。- 标记边界上的 'O':
- 遍历矩阵的边界(第一行、最后一行、第一列、最后一列),对于每一个 'O',使用 DFS 将其标记为 'B'。
- DFS 方法:
- 检查当前单元格是否在矩阵范围内,并且是否是 'O'。
- 如果是 'O',将其标记为 'B'。
- 递归处理当前单元格的上下左右四个方向。
- 替换和还原:
- 遍历整个矩阵,将所有的 'O' 替换为 'X'(这些是被围绕的区域)。
- 将所有的 'B' 还原为 'O'(这些是边界上的或连接到边界的 'O')。
import java.util.Arrays;
public class no_130 {
public static void main(String[] args) {
char[][] board = {
{'X', 'X', 'X', 'X'},
{'X', 'O', 'O', 'X'},
{'X', 'X', 'O', 'X'},
{'X', 'O', 'X', 'X'}
};
solve(board);
for (char[] chars : board) {
System.out.println(Arrays.toString(chars));
}
}
public static void solve(char[][] board) {
if (board == null || board.length == 0) {
return;
}
int m = board.length;
int n = board[0].length;
for (int i = 0; i < m; i++) {
dfs(board, i, 0);
dfs(board, i, n - 1);
}
for (int j = 0; j < n; j++) {
dfs(board, 0, j);
dfs(board, m - 1, j);
}
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
if (board[i][j] == 'O') {
board[i][j] = 'X';
} else if (board[i][j] == 'B') {
board[i][j] = 'O';
}
}
}
}
private static void dfs(char[][] board, int i, int j) {
int m = board.length;
int n = board[0].length;
if (i < 0 || i >= m || j < 0 || j >= n || board[i][j] != 'O') {
return;
}
board[i][j] = 'B';
dfs(board, i - 1, j);
dfs(board, i + 1, j);
dfs(board, i, j - 1);
dfs(board, i, j + 1);
}
}
本题关键:只要这个区域有一个点在边界上,那么这个区域肯定不能被围绕,除了边界的区域其他所有区域都是被围绕的,改为X。


















![24-7-6-读书笔记(八)-《蒙田随笔集》[法]蒙田 [译]潘丽珍](https://i-blog.csdnimg.cn/direct/bc0c8a58870d4d159a0d9a7661978021.jpeg)