【题目】289. 生命游戏
根据 百度百科 ,生命游戏,简称为生命,是英国数学家约翰·何顿·康威在 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]
]
进阶:
你可以使用原地算法解决本题吗?请注意,面板上所有格子需要同时被更新:你不能先更新某些格子,然后使用它们的更新后的值再更新其他格子。
本题中,我们使用二维数组来表示面板。原则上,面板是无限的,但当活细胞侵占了面板边界时会造成问题。你将如何解决这些问题?
【解题思路1】方向数组
因为是同时更新,所以为了防止更新后的值影响判断,讲board数组复制一份,用复制的数组作为判断依据进行判断
class Solution {
public void gameOfLife(int[][] board) {
int i = 0, j = 0, k = 0;
int m = board.length, n = board[0].length;
int[][] res = new int[m][n];
for(i=0; i<m; i++){
for(j=0; j<n; j++){
res[i][j] = board[i][j];
}
}
//方向数组,上下左右,右上、左上、右下、左下
int[] dx = {0, 0, -1, 1, -1, 1, -1, 1};
int[] dy = {-1, 1, 0, 0, -1, -1, 1, 1};
int count = 0;
for(i=0; i<m; i++){
for(j=0; j<n; j++){
count = 0;
for(k=0; k<8; k++){
int x = i+dx[k];
int y = j+dy[k];
if(x<0 || x>=m || y<0 || y>=n){
continue;
}
if(res[x][y]==1){
count++;
}
}
if(res[i][j] == 0 && count == 3){
board[i][j] = 1;
}
if(res[i][j] == 1 && (count<2 || count>3) ){
board[i][j] = 0;
}
}
}
}
}
【解题思路2】复合状态
定义复合状态 2,表示这个细胞现在是活的,之前是死的;
定义复合状态 -1,表示这个细胞现在是死的,之前是活的。
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]);
// Math.abs(x) //函数返回指定数字 “x“ 的绝对值
if ((r < rows && r >= 0) && (c < cols && c >= 0) && (Math.abs(board[r][c]) == 1)) {
liveNeighbors += 1;
}
}
}
}
if ((board[row][col] == 1) && (liveNeighbors < 2 || liveNeighbors > 3)) {
// -1 代表这个细胞过去是活的现在死了
board[row][col] = -1;
}
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;
}
}
}
}
}