题目:
给你一个 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。