TheAlgorithms-Python/cellular_automata/game_of_life.py

130 lines
3.0 KiB
Python
Raw Permalink Normal View History

"""Conway's Game Of Life, Author Anurag Kumar(mailto:anuragkumarak95@gmail.com)
Requirements:
- numpy
- random
- time
- matplotlib
Python:
- 3.5
Usage:
- $python3 game_of_life <canvas_size:int>
Game-Of-Life Rules:
1.
Any live cell with fewer than two live neighbours
dies, as if caused by under-population.
2.
Any live cell with two or three live neighbours lives
on to the next generation.
3.
Any live cell with more than three live neighbours
dies, as if by over-population.
4.
Any dead cell with exactly three live neighbours be-
comes a live cell, as if by reproduction.
2019-10-05 13:14:13 +08:00
"""
import random
import sys
import numpy as np
from matplotlib import pyplot as plt
from matplotlib.colors import ListedColormap
usage_doc = "Usage of script: script_name <size_of_canvas:int>"
2019-10-05 13:14:13 +08:00
choice = [0] * 100 + [1] * 10
random.shuffle(choice)
2019-10-05 13:14:13 +08:00
def create_canvas(size: int) -> list[list[bool]]:
2019-10-05 13:14:13 +08:00
canvas = [[False for i in range(size)] for j in range(size)]
return canvas
2019-10-05 13:14:13 +08:00
def seed(canvas: list[list[bool]]) -> None:
2019-10-05 13:14:13 +08:00
for i, row in enumerate(canvas):
for j, _ in enumerate(row):
canvas[i][j] = bool(random.getrandbits(1))
def run(canvas: list[list[bool]]) -> list[list[bool]]:
"""
This function runs the rules of game through all points, and changes their
status accordingly.(in the same canvas)
@Args:
--
canvas : canvas of population to run the rules on.
@returns:
--
canvas of population after one step
2019-10-05 13:14:13 +08:00
"""
current_canvas = np.array(canvas)
next_gen_canvas = np.array(create_canvas(current_canvas.shape[0]))
for r, row in enumerate(current_canvas):
for c, pt in enumerate(row):
2019-10-05 13:14:13 +08:00
next_gen_canvas[r][c] = __judge_point(
pt, current_canvas[r - 1 : r + 2, c - 1 : c + 2]
2019-10-05 13:14:13 +08:00
)
return next_gen_canvas.tolist()
2019-10-05 13:14:13 +08:00
def __judge_point(pt: bool, neighbours: list[list[bool]]) -> bool:
2019-10-05 13:14:13 +08:00
dead = 0
alive = 0
# finding dead or alive neighbours count.
for i in neighbours:
for status in i:
2019-10-05 13:14:13 +08:00
if status:
alive += 1
else:
dead += 1
# handling duplicate entry for focus pt.
2019-10-05 13:14:13 +08:00
if pt:
alive -= 1
else:
dead -= 1
# running the rules of game here.
state = pt
if pt:
2019-10-05 13:14:13 +08:00
if alive < 2:
state = False
elif alive in {2, 3}:
2019-10-05 13:14:13 +08:00
state = True
elif alive > 3:
state = False
else:
2019-10-05 13:14:13 +08:00
if alive == 3:
state = True
return state
2019-10-05 13:14:13 +08:00
if __name__ == "__main__":
if len(sys.argv) != 2:
raise Exception(usage_doc)
canvas_size = int(sys.argv[1])
# main working structure of this module.
2019-10-05 13:14:13 +08:00
c = create_canvas(canvas_size)
seed(c)
fig, ax = plt.subplots()
2019-10-05 13:14:13 +08:00
fig.show()
cmap = ListedColormap(["w", "k"])
try:
while True:
2019-10-05 13:14:13 +08:00
c = run(c)
ax.matshow(c, cmap=cmap)
fig.canvas.draw()
2019-10-05 13:14:13 +08:00
ax.cla()
except KeyboardInterrupt:
# do nothing.
pass