It was a lot of fun and we thought and discussed a lot about the problem. Thanks to all the participants and the organizers for the great time.
When I came back to Barcelona I decided to go on practicing by developing a full solution to Conway's Game of Life on my own.
I wanted the solution to include several of the ideas that we had been discussing during the code retreat:
- Tracking only living cells.
- Open/Closed with regard to the rules based on the number of neighbors.
- Open/Closed with regard to the number and location of a cell's neighbors.
- Open/Closed with regard to the dimensions of the grid (2D, 3D, 4D, etc).
It took me several rewrites but finally I came up with this solution that respects all the previous ideas.
- Tracking only living cells
A Generation has a collection of cells that are alive, LivingCells, and a set of Rules.
This way the "state of the cell" concept we tried in some iterations of the code retreat becomes pointless and it requires less space to store the current generation.
I also made Generation immutable.
To generate the next Generation of LivingCells (produceNextGeneration method), it first adds the surviving cells (addSurvivors method) and then the newly born cells (addNewCells method).
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package game; | |
public class Generation { | |
private LivingCells livingCells; | |
private Rules rules; | |
public Generation(LivingCells cells, Rules rules) { | |
this.livingCells = cells; | |
this.rules = rules; | |
} | |
public boolean isExtinct() { | |
return livingCells.empty(); | |
} | |
public Generation produceNextGeneration() { | |
LivingCells next = new LivingCells(); | |
addSurvivors(next); | |
addNewCells(next); | |
return new Generation(next, rules); | |
} | |
private void addSurvivors(LivingCells next) { | |
for (Cell cell : livingCells) { | |
if (willSurvive(cell)) | |
next.add(cell); | |
} | |
} | |
private void addNewCells(LivingCells next) { | |
Cells notAliveNeighbors = livingCells.getNotAliveNeighbors(); | |
for (Cell cell : notAliveNeighbors) { | |
if (willBeBorn(cell)) | |
next.add(cell); | |
} | |
} | |
private boolean willSurvive(Cell cell) { | |
return rules.shouldStayAlive(livingCells | |
.getAliveNeighborsNumberFor(cell)); | |
} | |
private boolean willBeBorn(Cell cell) { | |
return rules.shouldBeBorn(livingCells | |
.getAliveNeighborsNumberFor(cell)); | |
} | |
@Override | |
public String toString() { | |
return "[" + livingCells.toString() + "]"; | |
} | |
@Override | |
public boolean equals(Object obj) { | |
if (this == obj) | |
return true; | |
if (obj == null) | |
return false; | |
if (getClass() != obj.getClass()) | |
return false; | |
Generation other = (Generation) obj; | |
if (livingCells == null) { | |
if (other.livingCells != null) | |
return false; | |
} else if (!livingCells.equals(other.livingCells)) | |
return false; | |
return true; | |
} | |
} |
- Open/Closed with regard to the rules based on the number of neighbors.
Rules is an interface with two methods, shouldStayAlive and shouldBeBorn, that you can implement the way you wish.
The rules are still based on the number of neighbors which is the parameter passed to both methods.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package game; | |
public interface Rules { | |
public boolean shouldStayAlive(int numberOfNeighbors); | |
public boolean shouldBeBorn(int numberOfNeighbors); | |
} |
- Open/Closed with regard to the number and location of a cell's neighbors and Open/Closed with regard to the dimensions of the grid (2D, 3D, 4D, etc).
This two are possible thanks to the Cell interface. It has only a method, getNeighbors, which returns the neighbors of a cell. It's up to the cell to know which are its neighbors.
In this way each implementation of Cell can have a different dimension and a different number of neighbors which can be located following different stencils.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package game; | |
public interface Cell { | |
public Cells getNeighbors(); | |
} |
You can have a look at the code on GitHub.
Update: I refactored this code making it much simpler. Take a look at the new version.
No comments:
Post a Comment