首页 > 其他分享 >289. 生命游戏

289. 生命游戏

时间:2024-12-18 21:31:30浏览次数:7  
标签:状态 生命 游戏 int 细胞 289 board col row

生命游戏
根据 百度百科 , 生命游戏 ,简称为 生命 ,是英国数学家约翰·何顿·康威在 1970 年发明的细胞自动机。

给定一个包含 m × n 个格子的面板,每一个格子都可以看成是一个细胞。每个细胞都具有一个初始状态: 1 即为 活细胞 (live),或 0 即为 死细胞 (dead)。每个细胞与其八个相邻位置(水平,垂直,对角线)的细胞都遵循以下四条生存定律:

  1. 如果活细胞周围八个位置的活细胞数少于两个,则该位置活细胞死亡;

  2. 如果活细胞周围八个位置有两个或三个活细胞,则该位置活细胞仍然存活;

  3. 如果活细胞周围八个位置有超过三个活细胞,则该位置活细胞死亡;

  4. 如果死细胞周围正好有三个活细胞,则该位置死细胞复活;

下一个状态是通过将上述规则同时应用于当前状态下的每个细胞所形成的,其中细胞的出生和死亡是 同时 发生的。给你 m x n 网格面板 board 的当前状态,返回下一个状态。

给定当前 board 的状态,更新 board 到下一个状态。

注意 你不需要返回任何东西。

示例 1:
image

输入:board = [[0,1,0],[0,0,1],[1,1,1],[0,0,0]]
输出:[[0,0,0],[1,0,1],[0,1,1],[0,1,0]]

示例 2:
image

输入:board = [[1,1],[1,0]]
输出:[[1,1],[1,1]]

思路:

这个问题的关键是如何在不影响其他细胞状态的情况下,计算每个细胞的新状态。我们需要采取一种策略,在计算过程中避免直接修改当前状态,造成无法预料的影响。

为了实现这个思路,通常有两个步骤:

  1. 先计算出所有细胞的未来状态,但不立即更新棋盘的状态,而是用一个临时值来标记细胞的未来状态。
  2. 遍历整个棋盘,最终更新所有细胞的状态

在代码中,我们用 -12 作为临时标记来表示将要改变的状态:

  • -1:表示当前是活细胞(1),但是在下一代将会死。
  • 2:表示当前是死细胞(0),但是在下一代将会复活。

详细解释代码思路

  1. 邻居的遍历
    • 每个细胞有 8 个邻居。为了计算一个细胞周围的活邻居数量,我们可以遍历该细胞周围的 8 个方向,统计有多少个邻居是活细胞。
    • 我们用 neighbors = {-1, 0, 1} 来表示相邻的位置,分别表示上、下、左、右以及四个对角线方向。通过遍历这些方向,可以获取每个细胞的 8 个邻居的坐标。
  2. 规则判断和临时标记
    • 对于每个细胞,首先计算它周围活细胞的数量 liveNeighbors
    • 然后,根据以下规则来决定细胞的下一代状态:
      • 规则 1:如果细胞当前是活的(board[row][col] == 1),且它周围的活邻居小于 2 或者大于 3(即“孤独”或者“过度拥挤”),则该细胞会死。因此,我们将其标记为 -1,表示它将会死。
      • 规则 2:如果细胞当前是死的(board[row][col] == 0),且它周围有恰好 3 个活邻居(即“繁殖”),则该细胞将复活。因此,我们将其标记为 2,表示它将会活。
      • 规则 3:如果细胞当前是活的,并且它周围的活邻居数量正好为 2 或 3,那么它将继续存活,因此不需要标记。
  3. 更新棋盘状态
    • 一旦我们遍历完所有的细胞并根据规则标记了它们的状态,接下来就要更新棋盘了。
    • 在这一阶段,我们将所有 -1 转换为 0(死细胞),将所有 2 转换为 1(活细胞)。这样,我们就完成了所有细胞的状态更新。

步骤总结

  1. 遍历每个细胞,并计算该细胞周围的活邻居数量。
  2. 根据生命游戏规则,对每个细胞做出改变,使用临时标记 -12 来表示即将发生的状态变化。
  3. 遍历棋盘一次,将临时标记转换为实际的状态(即 -1 转换为 02 转换为 1)。

优点与难点

  • 避免直接修改原始状态:通过使用临时标记(-12),避免了直接修改原始状态对其他细胞产生不必要的影响。
  • 时间复杂度:该算法的时间复杂度是 O(m * n),其中 mn 分别是棋盘的行数和列数,因为每个细胞都需要被访问一次,且每个细胞有固定的邻居数量(8个)。
class Solution {
    public void gameOfLife(int[][] board) {
        int[] neighbors = {-1, 0, 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)){
                            //相邻位置的坐标
                            //当 i = -1 和 j = 1 时,r = row - 1,c = col + 1,即左上方的邻居。
                            //当 i = 1 和 j = 0 时,r = row + 1,c = col,即下方的邻居。
                            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 或 规则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;
                }
            }
        }
        //整体遍历一遍更新状态
        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;
                }
            }
        }
    }
}

标签:状态,生命,游戏,int,细胞,289,board,col,row
From: https://www.cnblogs.com/drunkerl/p/18615878

相关文章

  • 534. 游戏玩法分析 III - 力扣(LeetCode)
    534.游戏玩法分析III-力扣(LeetCode)目标输入输入:Activitytable:player_iddevice_idevent_dategames_played122016/3/15122016/5/26132017/6/251312016/3/20342018/7/35输出输出:player_idevent_dategames_played_so_far12016/3/1512016/5/21112017/6/251232016/3/......
  • 《游戏设计艺术(第三版)》笔记
    《游戏设计艺术(TheArtofGameDesign)》第三版[美]JesseSchell著,刘嘉俊杨逸等译电子工业出版社目录第1章 太初之时,有设计师第2章设计师创造体验第3章体验发生于场景第4章体验从游戏中诞生第5章游戏由元素构成第6章元素支撑起主题第7章游戏始于一个创意......
  • 学霸带你游戏化理解 Python 装饰器与生成器
    使代码更高效在现代Python开发中,装饰器和生成器为编程提供了强大的功能和灵活性。它们不仅能帮助程序员优化性能,还能使代码结构更加简洁、清晰。装饰器通过增强现有函数的功能,使得代码更具可扩展性和重用性;生成器则通过惰性求值提高内存效率,让你能够高效处理大规模数据。无......
  • 学霸带你游戏化训练批判性阅读打破传统思维模式
    高效的阅读技巧在面对海量信息时,提高阅读效率和准确性至关重要。批判性阅读不仅要求我们具备深入分析的能力,还需要有效管理阅读的时间和精力。通过提升阅读技巧,我们能够快速抓住关键信息,并高效完成信息筛选与处理。提升扫描与浏览技巧批判性阅读中,快速扫描和浏览能帮助我们......
  • 冬季限定节日游戏工作流优化,6 款办公软件哪家强?
    在游戏行业的节日盛宴中,每一个新游戏的上线或重大更新都是一场与时间赛跑的挑战,需要开发团队、测试团队和运营团队如同精密齿轮般紧密协作。而可视化团队协作办公软件则成为了推动这一复杂机器高效运转的润滑剂。本文将站在全J人游戏公司的视角,深入剖析6款此类办公软件,探讨它......
  • 冬季游戏协作挑战,6 款办公软件能否成为团队的坚实后盾?
    在游戏行业的节日盛宴中,每一个新游戏的上线或重大更新都是一场与时间赛跑的挑战,需要开发团队、测试团队和运营团队如同精密齿轮般紧密协作。而可视化团队协作办公软件则成为了推动这一复杂机器高效运转的润滑剂。本文将站在全J人游戏公司的视角,深入剖析6款此类办公软件,探讨它......
  • 游戏团队节日协作攻略!6 款办公软件怎样成为致胜法宝?
    在游戏行业这个充满创意与挑战的领域,各部门间的紧密协作是项目成功的基石,尤其是在节日期间,新游戏上线或重大更新的压力下,高效的办公软件成为了不可或缺的协作利器。本文将站在全J人游戏公司的视角,深入剖析6款可视化团队协作办公软件,探讨它们在游戏开发、运营等多方面的表现,究......
  • 游戏各部门节日协作,6 款办公软件如何引发效率革命?
    在游戏行业的激烈竞争中,冬季作为重要的营销节点,众多游戏公司都力争在这个时期推出新游戏或重大更新,以吸引玩家的目光并抢占市场份额。这一过程犹如一场紧张的冲刺赛,需要开发团队、测试团队、运营团队等多部门密切协作,高效的办公软件则成为了团队加速前进的助推器。本文将站在全J......
  • 冬季游戏高峰,哪 6 款办公软件可让团队协作畅通无阻?
    在游戏行业这个充满创意与挑战的领域,尤其是在节日期间,新游戏上线和重大更新任务紧迫,开发团队、测试团队与运营团队之间的高效协作成为了项目成功的关键。一款优秀的可视化团队协作办公软件能够如同精密的齿轮组,带动整个团队顺畅运转,板栗看板便是其中的佼佼者。接下来,让我们一同盘......
  • 冬季游戏上线冲刺,哪 6 款办公软件可助团队协作效率飙升?
    在游戏行业中,尤其是节假日期间,团队的协作效率直接影响到项目的进度和质量。对于开发、测试、运营等多个职能部门,只有紧密合作才能确保新游戏的上线或重大更新按时完成,并为玩家提供最佳的游戏体验。在这一过程中,合适的团队协作工具显得尤为重要。板栗看板作为一款可视化的任务管理......