【问题标题】:Conway's Game of Life not counting neighbors correctly康威的生命游戏没有正确计算邻居数
【发布时间】:2016-11-02 05:45:48
【问题描述】:

我正在使用 Python 执行标准的康威生命游戏程序。我在遍历数组时尝试计算邻居时遇到问题。我创建了打印语句来打印 if 语句的连续性,以及每个语句的计数值。

这是我的代码:(我在整个代码的 # 中都有问题)

import random

numrows = 10
numcols = 10
def rnd():
    rn = random.randint(0,1)
    return rn

def initial():
    grid = []
    count = 0
    for x in range(numrows):
        grid.append([])
        for y in range(numcols):
            rand=random.randrange(1,3)
            if(rand == 1):
                grid[x].append('O')
            else:
                grid[x].append('-')
    for x in grid:    
        print(*x, sep=' ',end="\n") #this prints the random 2d array 

    print("")
    print("")

    answer = 'y'
    newgrid = []
    count = 0

    while(answer == 'y'): # I believe I am going through, checking neighbors
                          # and moving onto the next index inside these for 
                          #loops below
        for r in range(0,numrows):
            grid.append([])
            for c in range(0,numcols):

                if(r-1 > -1 and c-1 > -1): #I use this to check out of bound
                    if(newgrid[r-1][c-1] == 'O'):#if top left location is O
                        count = count + 1        #should get count += 1
                    else:
                        count = count
                print("top left check complete")
                print(count)
                if(r-1 > -1):
                    if(newgrid[r-1][c] == 'O'):
                        count = count + 1
                    else:
                        count = count
                print("top mid check complete")
                print(count)
                if(r-1 > -1 and c+1 < numcols):
                    if(newgrid[r-1][c+1] == 'O'):
                        count = count + 1
                    else:
                        count = count
                print("top right check complete")
                print(count)
                if(c-1 > -1 and r-1 > -1):
                    if(newgrid[r][c-1] == 'O'):
                        count = count + 1
                    else:
                        count = count
                print("mid left check complete")
                print(count)
                if(r-1 > -1 and c+1 < numcols): 
                    if(newgrid[r][c+1] == 'O'):
                        count = count + 1
                    else:
                        count = count
                print("mid right check complete")
                print(count)
                if(r+1 < numrows and c-1 > -1):
                    if(newgrid[r+1][c-1] == 'O'):
                        count = count + 1
                    else:
                        count = count
                print("bot left check complete")
                print(count)
                if(r+1 < numrows and c-1 > -1):
                    if(newgrid[r+1][c] == 'O'):
                        count = count + 1
                    else:
                        count = count
                print("bot mid check complete")
                print(count)
                if(r+1 < numrows and c+1 < numcols):
                    if(newgrid[r+1][c+1] == 'O'):
                        count = count + 1
                    else:
                        count = count
                print("bot right check complete")
                print(count)

# I am not sure about the formatting of the code below, how do I know that
# the newgrid[r][c] location is changing? should it be according to the for-
# loop above? Or should it get it's own? If so, how could I construct it as
# to not interfere with the other loops and items of them?


            if(newgrid[r][c] == '-' and count == 3):
                newgrid[r][c] ='O'

            elif(newgrid[r][c] == 'O' and count < 2):
                newgrid[r][c] = '-'

            elif(newgrid[r][c] == 'O' and (count == 2 or count == 3)):
                newgrid[r][c] = 'O'

            elif(newgrid[r][c] == 'O' and count > 3):
                newgrid[r][c] = '-'

# I'm also confused how to go about printing out the 'new' grid after each
# element has been evaluated and changed. I do however know that after the
# new grid prints, that I need to assign it to the old grid, so that it can
# be the 'new' default grid. How do I do this? 


        for z in newgrid:
            print(*z, sep=' ',end="\n")


        answer = input("Continue? y or n( lower case only): ")

        newgrid = grid

        if(answer != 'y'):
            print(" Hope you had a great life! Goodbye!")

initial()

这是当前的输出和错误信息:

>>> initial()
 - O - - O - - O - -
 - O - - O - - - O O
 - O - - O - O O - O
 O - - O - - O O O O
 O - O O - - - O O -
 O - O - O - O - O -
 O - O O O O - - O -
 - - - - O O O - - O
 O O - O - - O - - -
 - - O O O - O - - -
top left check complete
0
top mid check complete
0
top right check complete
0
mid left check complete
0
mid right check complete
0
bot left check complete
0
bot mid check complete
0
Traceback (most recent call last):
 File "<pyshell#68>", line 1, in <module>
  initial()
 File "C:\Users\Ted\Desktop\combined.py", line 86, in initial
  if(newgrid[r+1][c+1] == 'O'):
 IndexError: list index out of range

当我遍历随机数组以查看邻居是什么时,它似乎没问题,直到它移动到 [0][1] 同时检查机器人右邻居。

另外,中间的右边邻居应该 +1 才能算作是活着的。但是,即使 if 语句连续,count 仍然是 0?

问题 1:我怎么可能知道我的 if 条件足以满足数组所有边的 [r][c] 的每个实例?

问题 2:我目前的检查越界方法是否最适合我的情况?有没有办法在我检查值之前“检查所有是否越界”?

此时我已经束手无策了。提前感谢您花时间帮助回答我的问题

【问题讨论】:

  • 一个建议是使用调试器并单步执行代码。
  • 看起来错误发生位置附近的很多代码都有gridnewgrid 切换。 newgrid 是空的,但是您正在向 grid 添加行,并且您正在 newgrid 中查找内容(导致异常)。
  • @Blckknght 好点!我原本打算在回答中提到这一点,但我被其他问题分心了。我现在已经添加了该信息,所以谢谢你提醒我。

标签: python arrays conways-game-of-life


【解决方案1】:

您收到该索引错误是因为您的 newgrid 仅包含一个空行。并且您在 newgrid 而不是 grid 中对邻居进行测试(正如 Blckknght 在 cmets 中提到的那样)。我已经进行了一些修复,但是还有很多可以改进此代码的工作。看起来它现在正在工作,但很难判断你何时使用随机生命形式。 :) 我建议给你的程序一些使用已知 Life 模式的方法,比如闪烁器和滑翔机,以查看它们的行为是否正确。

确保newgrid 有效的最简单方法是从grid 复制它。如果我们只做newgrid = grid,那只会使newgrid 成为grid 对象的另一个名称。要正确复制列表列表,我们需要复制每个内部列表。我的新代码使用 copy_grid 函数来实现这一点。

我已经修复了您在计算邻居部分的if 测试中遇到的几个小错误,并且我已经简化了根据邻居计数更新单元格的逻辑。我还压缩了生成随机网格的代码,并添加了一个简单的函数,它可以从字符串中读取生命模式并从中构建网格。这让我们可以使用 Glider 测试代码。我还添加了一个创建空网格的函数。尽管我在测试期间使用了该程序,但该程序当前不使用该功能,我想这是一个有用的示例。 :)

import random

# Set a seed so that we get the same random numbers each time we run the program
# This makes it easier to test the program during development
random.seed(42)

numrows = 10
numcols = 10

glider = '''\
----------
--O-------
---O------
-OOO------
----------
----------
----------
----------
----------
----------
'''

# Make an empty grid
def empty_grid():
    return [['-' for y in range(numcols)]
        for x in range(numrows)]

# Make a random grid
def random_grid():
    return [[random.choice('O-') for y in range(numcols)]
        for x in range(numrows)]

# Make a grid from a pattern string
def pattern_grid(pattern):
    return [list(row) for row in pattern.splitlines()]

# Copy a grid, properly!
def copy_grid(grid):
    return [row[:] for row in grid]

# Print a grid
def show_grid(grid):
    for row in grid:
        print(*row)
    print()

def run(grid):
    show_grid(grid)

    # Copy the grid to newgrid.
    newgrid = copy_grid(grid)
    while True:
        for r in range(numrows):
            for c in range(numcols):
                # Count the neighbours, making sure that they are in bounds
                count = 0
                # Above this row
                if(r-1 > -1 and c-1 > -1): 
                    if(grid[r-1][c-1] == 'O'):
                        count += 1
                if(r-1 > -1):
                    if(grid[r-1][c] == 'O'):
                        count += 1
                if(r-1 > -1 and c+1 < numcols):
                    if(grid[r-1][c+1] == 'O'):
                        count += 1

                # On this row
                if(c-1 > -1):
                    if(grid[r][c-1] == 'O'):
                        count += 1
                if(c+1 < numcols): 
                    if(grid[r][c+1] == 'O'):
                        count += 1

                # Below this row
                if(r+1 < numrows and c-1 > -1):
                    if(grid[r+1][c-1] == 'O'):
                        count += 1
                if(r+1 < numrows):
                    if(grid[r+1][c] == 'O'):
                        count += 1
                if(r+1 < numrows and c+1 < numcols):
                    if(grid[r+1][c+1] == 'O'):
                        count += 1

                # Update the cell in the new grid
                if grid[r][c] == '-':
                    if count == 3:
                        newgrid[r][c] ='O'
                else:
                    if count < 2 or count> 3:
                        newgrid[r][c] = '-'

        # Copy the newgrid to grid
        grid = copy_grid(newgrid)
        show_grid(grid)

        answer = input("Continue? [Y/n]: ")
        if not answer in 'yY':
            print(" Hope you had a great life! Goodbye!")
            break

#grid = random_grid()
grid = pattern_grid(glider)
run(grid)

此代码确实可以正常工作,但仍有大量的改进空间。例如,这是run() 的改进版本,它通过使用几个循环来压缩邻居计数部分。

def run(grid):
    show_grid(grid)

    # Copy the grid to newgrid.
    newgrid = copy_grid(grid)
    while True:
        for r in range(numrows):
            for c in range(numcols):
                # Count the neighbours, making sure that they are in bounds
                # This includes the cell itself in the count
                count = 0
                for y in range(max(0, r - 1), min(r + 2, numrows)):
                    for x in range(max(0, c - 1), min(c + 2, numcols)):
                        count += grid[y][x] == 'O'

                # Update the cell in the new grid
                if grid[r][c] == '-':
                    if count == 3:
                        newgrid[r][c] ='O'
                else:
                    # Remember, this count includes the cell itself
                    if count < 3 or count > 4:
                        newgrid[r][c] = '-'

        # Copy the newgrid to grid
        grid = copy_grid(newgrid)
        show_grid(grid)

        answer = input("Continue? [Y/n]: ")
        if not answer in 'yY':
            print(" Hope you had a great life! Goodbye!")
            break

【讨论】:

  • 非常感谢!你帮助我在一个晚上比我整个星期走得更远!感谢您抽出时间更正我的代码。我将把这条线与我的线进行比较,并了解我出错的所有地方。再次感谢!!
  • 我无法告诉你我多么感谢你抽出时间来帮助我。你超越了,我不能感谢你。我在这个项目上被小事缠住了。与我的代码相比,我已经研究了这段代码,并且已经学到了很多东西。你是一个救生员!
  • @Ted 感谢您接受我的回答。我必须承认,我以前的版本中存在基本列表复制错误有点尴尬,所以我觉得有义务确保新版本正常工作。 :)
【解决方案2】:

要计算邻居,只需遵循这个简单的规则。您将拥有当前单元格的行和列。使用这种形式的两个数组,一个用于行,另一个用于列。以下是您的逻辑。

您可以通过演示视频here找到详细实现。

this.neibhours = function() {
  var rows = [row-1, row, row+1];
  var cols = [col-1, col, col+1];
  neibhourCells = [];

  for(var i=0; i < rows.length; i++) {
    for(var j=0; j < cols.length; j++) {
      if(!(col === cols[j] && row === rows[i])) {
        var cell = new Cell(rows[i], cols[j])
        if(cell.isOnGrid()) {
          neibhourCells.push(cell);
        }
      }
    }
  }

  return neibhourCells;
}

【讨论】:

    猜你喜欢
    • 2023-03-28
    • 2019-04-16
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-08-13
    • 1970-01-01
    • 2013-02-21
    相关资源
    最近更新 更多