A game of life🔬 simulator on an infinite♾️ plane
NOTE: This is a toy project! I did this just for fun, not as a packaged product.
I'm a bored sophomore in college working on projects to fill the time. If you enjoy my work, consider supporting me by buying me a coffee!
- Any live cell with fewer than two live neighbours dies, as if by starvation.
- Any live cell with two or three live neighbours lives on to the next generation.
- Any live cell with more than three live neighbours dies, as if by overpopulation.
- Any dead cell with exactly three live neighbours becomes a live cell, as if by reproduction.
When making a Game of Life simulator, though, most people just get the minimal, barebones simulation working on a glider or a few other patterns before finishing up the project, and more importantly, we only ever learn how to implement the Game on a statically sized grid. How do the more complicated simulators on the internet support theoretically infinitely sized worlds🧫???
Thinking about this got me curious, so I took a stab at it myself. I think I did an okay job!
The key to the solution is expressed in the wording of the problem itself: to make an implementation agnostic of infinitely sized worlds, the implementation must not depend on the world's size.
So what does that mean in practice? Well, it means that any implementation using statically allocated arrays for the world is already out of the running. But what about dynamically sized arrays? Although, yes, in theory these are possible, they would be incredibly inefficient. Every time a glider moved beyond a corner of the world, the program would have to allocate a new row, and a new column: which would compound as the glider keeps going! Your program would very quickly run out of memory, unless it could deallocate parts of the grid it wasn't using, which would take quite a bit of time on its own.
For some, it might be painfully obvious that the solution is to only store the live cells mapped from their position. So for a glider traversing the entire world for an infinitely long, your program only every stores data for 5 cells at once! This has the added benefit that this makes it very simple to only ever consider cells which are already neighbors of live cells.
The entire algorithm works as follows for each iteration:
- For every live cell, if it is overcrowded (>3 neighbors) or starving (<2 neighbors), remove it from the next iteration.
- Additionally, for every neighboring position of that cell (regardless of whether it was removed):
- If the position has already been considered, skip the following steps. Otherwise, add the position to the set of previously considered positions.
- If the position is overcrowded or starving, remove the cell at that position from the next iteration (if it exists).
- If the position has exactly three live neighbors, however, insert a cell at that position in the next iteration.
- Voila! You have the next iteration of the world!
To see it in action, run some of the examples!
To run, you must download Rust from here.
# Clone the repo and run from source
git clone https://github.com/adam-mcdaniel/game-of-life
cd game-of-life
cargo run --example random --release