-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconway.py
223 lines (182 loc) · 6.22 KB
/
conway.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
"""A simple implementation Conway's game of life to demonstrate sets and tuples in Python.
This implementation is intentionally kept naive for simplicity."""
__author__ = "Hovis <[email protected]>"
# CONSTANTS
# I'm bounding the world to keep logic and display trivial, but using
# a set for state allows practically unlimited board size!
BOUND = 15
DELAY = .3 #seconds
# GLOBALS (some pre-defined patterns)
spinner = set([(1, 0), (1, 1), (1, 2)])
glider = set([(9, 6), (9, 7), (9, 8), (10, 6), (11, 7)])
beehive = set([(1, 10), (2, 9), (2, 11), (3, 9), (3, 11), (4, 10)])
spinnersplosion = set([(6, 7), (7, 7), (8, 7), (7, 9)])
def _main(board=spinner.union(glider)):
import time
mainboard = board
# Draw stuff, break on Ctrl-C
while True:
draw_board(mainboard)
time.sleep(DELAY) #seconds
mainboard = iterate(mainboard)
def iterate(board):
""" Return the next generation of the passed board.
>>> iterate( set([(1,0), (1,1), (1,2)]) )
{(0, 1), (1, 1), (2, 1)}
"""
# Sets are mutable, if we edit it directly we'll screw things up because
# LIFE is supposed to iterate a whole universe at a time
next_board = set()
# Let's get our locations using a comprehension instead of nested loops.
# The comprehension evaluates to a generator that returns tuples.
# If we made a list from the generator, it would look like: [(0,0), (0,1), (0,2), ..., (14, 14)]
for location in ((x, y) for x in range(BOUND) for y in range(BOUND)):
# location is a 2-tuple that corresponds to an x, y coordinate
if cell_next_state( is_alive(location, board), neighbor_count(location, board) ):
next_board.add(location)
return next_board
def cell_next_state(alive, neighbors):
""" Return True or False depending on whether the given cell should be alive in the next generation.
Operates according to Conway's original rules (B3/S23).
These rules mean that an dead cell should become alive if it has 3 live neighbors,
and an alive cell should stay alive if it has 2 or 3 live neighbors, otherwise the live cell should die.
Examples:
>>> cell_next_state(alive=True, neighbors=2)
True
>>> cell_next_state(alive=True, neighbors=3)
True
>>> cell_next_state(alive=False, neighbors=3)
True
>>> cell_next_state(alive=False, neighbors=2)
False
>>> cell_next_state(alive=False, neighbors=4)
False
"""
# There are shorter ways of structuring this logic, but this more closely
# represents the description of the logic ("B3/S23")
if alive:
return neighbors in (2, 3)
#return neighbors == 2 or neighbors == 3
else:
return neighbors == 3
def neighbor_count(location, board):
""" Returns the number of neighbor cells that are "alive" (in the set).
>>> testboard = set([(0,0), (2,1)])
>>> neighbor_count((1,1), testboard)
2
The board is generated using a comprehension. Yay for expressiveness!
>>> testboard = set([(x, y) for x in (0,1,2) for y in (0,1,2)])
>>> neighbor_count((1,1), testboard)
8
"""
count = 0
for neighbor in get_neighbors(location):
if is_alive(neighbor, board): #If we don't pass the board through, the tests will fail. (more hacks because global state sucks)
count += 1
return count
def is_alive(location, board):
""" Return True if the cell in location is alive, False if dead
>>> testboard = set([(0,0), (1,1), (2,2)])
>>> location = 0, 0
>>> is_alive(location, testboard)
True
>>> location = 0, 1
>>> is_alive(location, testboard)
False
>>> location = (1, 1)
>>> is_alive(location, testboard)
True
>>> location = (2,0)
>>> is_alive(location, testboard)
False
"""
# Our board is a set of all the "alive" cell locations.
# Any cell locations not in the set are implicitly "dead".
return location in board
def get_neighbors(location, bound=BOUND):
"""Returns a tuple of locations that are immediately adjacent to the given location, wrapping around bounds.
+ : loc
O : expected neighbors
012
0 OOO
1 O+O
2 OOO
>>> get_neighbors((1,1))
[(0, 0), (0, 1), (0, 2), (1, 0), (1, 2), (2, 0), (2, 1), (2, 2)]
012345678901
0 +O OO
1 OO
2
3
4
5
6
7
8
9
0
1 OO O
>>> get_neighbors(location=(0,0), bound=12)
[(11, 11), (11, 0), (11, 1), (0, 11), (0, 1), (1, 11), (1, 0), (1, 1)]
"""
# I'm going to use a mutable list to keep track of my neighbors, and then return it
# This can be optimized with a generator, see neighbors_generator
x, y = location
neighbors = []
for offset_x in (-1, 0, 1):
for offset_y in (-1, 0, 1):
nx = (x + offset_x) % bound
ny = (y + offset_y) % bound
if (nx, ny) == location:
continue # The location passed to the function is not a neighbor
neighbors.append((nx, ny))
return neighbors
def neighbors_generator(location, bound=BOUND):
"""
Like get_neighbors(), but return a generator instead
For testing, we'll pass the generator to tuple() to see what it yields.
X : loc
O : expected neighbors
012
0 OOO
1 O+O
2 OOO
>>> loc = 1, 1
>>> tuple(neighbors_generator(loc))
((0, 0), (0, 1), (0, 2), (1, 0), (1, 2), (2, 0), (2, 1), (2, 2))
012345678901
0 +O OO
1 OO
2
3
4
5
6
7
8
9
0
1 OO O
>>> loc = 0, 0
>>> tuple(neighbors_generator(loc, bound=12))
((11, 11), (11, 0), (11, 1), (0, 11), (0, 1), (1, 11), (1, 0), (1, 1))
"""
x, y = location
return (
( (x + offset_x) % bound, (y + offset_y) % bound ) # tuple specification
for offset_x in (-1, 0, 1)
for offset_y in (-1, 0, 1)
if not (offset_x == 0 and offset_y == 0) # exclude the center
)
def _test():
import doctest
print(doctest.testmod())
def draw_board(board):
for x in range(BOUND):
for y in range(BOUND):
print( "@" if is_alive((x,y), board) else "`", end='')
print() #need that newline
print("\n"*2)
if __name__ == "__main__":
import sys
sys.exit(_main())