难度中等
根据 百度百科 ,生命游戏,简称为生命,是英国数学家约翰·何顿·康威在 1970 年发明的细胞自动机。
给定一个包含 m × n 个格子的面板,每一个格子都可以看成是一个细胞。每个细胞都具有一个初始状态:1 即为活细胞(live),或 0 即为死细胞(dead)。每个细胞与其八个相邻位置(水平,垂直,对角线)的细胞都遵循以下四条生存定律:
根据当前状态,写一个函数来计算面板上所有细胞的下一个(一次更新后的)状态。下一个状态是通过将上述规则同时应用于当前状态下的每个细胞所形成的,其中细胞的出生和死亡是同时发生的。
示例:
输入:
[
[0,1,0],
[0,0,1],
[1,1,1],
[0,0,0]
]
输出:
[
[0,0,0],
[1,0,1],
[0,1,1],
[0,1,0]
]
进阶:
在讲具体解法之前,请先根据下面的图片理解题目中描述的细胞遵循的生存定律,这有助于我们后面的讲解。
思路
这个问题看起来很简单,但有一个陷阱,如果你直接根据规则更新原始数组,那么就做不到题目中说的 同步 更新。假设你直接将更新后的细胞状态填入原始数组,那么当前轮次其他细胞状态的更新就会引用到当前轮已更新细胞的状态,但实际上每一轮更新需要依赖上一轮细胞的状态,是不能用这一轮的细胞状态来更新的。
如上图所示,已更新细胞的状态会影响到周围其他还未更新细胞状态的计算。一个最简单的解决方法就是复制一份原始数组,复制的那一份永远不修改,只作为更新规则的引用。这样原始数组的细胞值就不会被污染了。
算法
board
中的细胞状态。class Solution {
public void gameOfLife(int[][] board) {
int[] neighbors = {0, 1, -1};
int rows = board.length;
int cols = board[0].length;
// 创建复制数组 copyBoard
int[][] copyBoard = new int[rows][cols];
// 从原数组复制一份到 copyBoard 中
for (int row = 0; row < rows; row++) {
for (int col = 0; col < cols; col++) {
copyBoard[row][col] = board[row][col];
}
}
// 遍历面板每一个格子里的细胞
for (int row = 0; row < rows; row++) {
for (int col = 0; col < cols; col++) {
// 对于每一个细胞统计其八个相邻位置里的活细胞数量
int liveNeighbors = 0;
for (int i = 0; i < 3; i++) {
for (int j = 0; j < 3; j++) {
if (!(neighbors[i] == 0 && neighbors[j] == 0)) {
int r = (row + neighbors[i]);
int c = (col + neighbors[j]);
// 查看相邻的细胞是否是活细胞
if ((r < rows && r >= 0) && (c < cols && c >= 0) && (copyBoard[r][c] == 1)) {
liveNeighbors += 1;
}
}
}
}
// 规则 1 或规则 3
if ((copyBoard[row][col] == 1) && (liveNeighbors < 2 || liveNeighbors > 3)) {
board[row][col] = 0;
}
// 规则 4
if (copyBoard[row][col] == 0 && liveNeighbors == 3) {
board[row][col] = 1;
}
}
}
}
}
复杂度分析
board
的行数和列数。思路
方法一中 O(mn) 的空间复杂度在数组很大的时候内存消耗是非常昂贵的。题目中每个细胞只有两种状态 live(1)
或 dead(0)
,但我们可以拓展一些复合状态使其包含之前的状态。举个例子,如果细胞之前的状态是 0
,但是在更新之后变成了 1
,我们就可以给它定义一个复合状态 2
。这样我们看到 2
,既能知道目前这个细胞是活的,还能知道它之前是死的。
算法
board
中的细胞。-1
,代表这个细胞过去是活的现在死了;1
;-1
,代表这个细胞过去是活的现在死了。可以看到,因为规则 1 和规则 3 下细胞的起始终止状态是一致的,因此它们的复合状态也一致;2
,代表这个细胞过去是死的现在活了。board
转成 0
,1
的形式。因此这时候需要再遍历一次数组,将复合状态为 2
的细胞的值改为 1
,复合状态为 -1
的细胞的值改为 0
。class Solution {
public void gameOfLife(int[][] board) {
int[] neighbors = {0, 1, -1};
int rows = board.length;
int cols = board[0].length;
// 遍历面板每一个格子里的细胞
for (int row = 0; row < rows; row++) {
for (int col = 0; col < cols; col++) {
// 对于每一个细胞统计其八个相邻位置里的活细胞数量
int liveNeighbors = 0;
for (int i = 0; i < 3; i++) {
for (int j = 0; j < 3; j++) {
if (!(neighbors[i] == 0 && neighbors[j] == 0)) {
// 相邻位置的坐标
int r = (row + neighbors[i]);
int c = (col + neighbors[j]);
// 查看相邻的细胞是否是活细胞
if ((r < rows && r >= 0) && (c < cols && c >= 0) && (Math.abs(board[r][c]) == 1)) {
liveNeighbors += 1;
}
}
}
}
// 规则 1 或规则 3
if ((board[row][col] == 1) && (liveNeighbors < 2 || liveNeighbors > 3)) {
// -1 代表这个细胞过去是活的现在死了
board[row][col] = -1;
}
// 规则 4
if (board[row][col] == 0 && liveNeighbors == 3) {
// 2 代表这个细胞过去是死的现在活了
board[row][col] = 2;
}
}
}
// 遍历 board 得到一次更新后的状态
for (int row = 0; row < rows; row++) {
for (int col = 0; col < cols; col++) {
if (board[row][col] > 0) {
board[row][col] = 1;
} else {
board[row][col] = 0;
}
}
}
}
}
复杂度分析
board
的行数和列数。分析:11 代表细胞活的, 00 代表细胞死的,那么这个位置就四种状态,用【下一个状态,当前状态】表示,最后需要用右移操作更新结果
进一步:下一轮活的可能有两种,也就是要把单元格变为 1
那遍历下每个格子看他周围细胞有多少个活细胞就行了,然后更改为状态,那么对于第一种可能,把 board[i]设置为 3,对于第二种可能状态设置为 2,设置个高位flag,遍历后面的格子,拿到与他相邻的格子中有多少个 alive的,和 11 与一下即可,最后我们把 board[i]右移 11位,更新结果
具体代码,我写了注释~
class Solution {
int[] dx = {-1, 1, 0, 0, -1, -1, 1, 1};
int[] dy = {0, 0, -1, 1, -1, 1, -1, 1};
int[][] board;
int m, n;
public void gameOfLife(int[][] board) {
this.board = board;
// 特判
if (board == null || board.length == 0 || board[0] == null || board[0].length == 0) return;
this.m = board.length;
this.n = board[0].length;
// 遍历
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
// 拿到当前位置周围活细胞数量
int cnt = countAlive(i, j);
// 1. 活细胞周围八个位置有两个或三个活细胞,下一轮继续活
if (board[i][j] == 1 && (cnt == 2 || cnt == 3)) board[i][j] = 3;
// 2. 死细胞周围有三个活细胞,下一轮复活了
if (board[i][j] == 0 && cnt == 3) board[i][j] = 2;
}
}
// 更新结果
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
board[i][j] >>= 1;
}
}
}
private int countAlive(int x, int y) {
int cnt = 0;
for (int k = 0; k < 8; k++) {
int nx = x + dx[k];
int ny = y + dy[k];
if (nx < 0 || nx >= m || ny < 0 || ny >= n) continue;
// 如果这个位置为 0,代表当前轮是死的,不需要算进去
// 如果这个位置为 1,代表当前轮是活得,需要算进去
// 如果这个位置为 2,代表当前轮是死的(状态10,下一轮是活的),不需要算进去
// 如果这个位置为 3,代表是当前轮是活的(状态11,下一轮也是活的),需要算进去
cnt += (board[nx][ny] & 1);
}
return cnt;
}
}
) continue;
// 如果这个位置为 0,代表当前轮是死的,不需要算进去
// 如果这个位置为 1,代表当前轮是活得,需要算进去
// 如果这个位置为 2,代表当前轮是死的(状态10,下一轮是活的),不需要算进去
// 如果这个位置为 3,代表是当前轮是活的(状态11,下一轮也是活的),需要算进去
cnt += (board[nx][ny] & 1);
}
return cnt;
}
}