This is the UML for the refactored design:
I eliminated the LivingCells class that appeared in the previous design.
I pushed a lot of code from the GameOfLife class into Generation to eliminate a bit of feature envy. This is GameOfLife's code now:
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 GameOfLife { | |
private Generation currentGeneration; | |
public GameOfLife(Generation initialGeneration) { | |
this.currentGeneration = initialGeneration; | |
} | |
public void run(int steps) { | |
currentGeneration = evolve(steps, currentGeneration); | |
} | |
private Generation evolve(int steps, Generation currentGeneration) { | |
if (isOver(steps)) { | |
return currentGeneration; | |
} | |
return evolve( | |
steps - 1, | |
currentGeneration.produceNextGeneration()); | |
} | |
private boolean isOver(int steps) { | |
return steps == 0 || currentGeneration.extinct(); | |
} | |
public Generation currentGeneration() { | |
return currentGeneration; | |
} | |
} |
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 Rules rules; | |
private Cells aliveCells; | |
public Generation(Rules rules, Cell... aliveCells) { | |
this(rules, new Cells(aliveCells)); | |
} | |
public Generation(Rules rules, Cells aliveCells) { | |
this.rules = rules; | |
this.aliveCells = aliveCells; | |
} | |
public Generation produceNextGeneration() { | |
Cells nextGeneration = new Cells(); | |
for (Cell neighbor : aliveCells.getNeighbors()) { | |
if (inNextGeneration(neighbor)) { | |
nextGeneration.add(neighbor); | |
} | |
} | |
return new Generation(rules, nextGeneration); | |
} | |
private boolean inNextGeneration(Cell neighbor) { | |
return rules.inNextGeneration( | |
isAlive(neighbor), | |
aliveNeighborsNumberFor(neighbor)); | |
} | |
private int aliveNeighborsNumberFor(Cell cell) { | |
int res = 0; | |
for (Cell neighbor : cell.neighbors()) { | |
if (isAlive(neighbor)) { | |
res++; | |
} | |
} | |
return res; | |
} | |
private boolean isAlive(Cell cell) { | |
return aliveCells.contains(cell); | |
} | |
public boolean extinct() { | |
return aliveCells.empty(); | |
} | |
@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 (aliveCells == null) { | |
if (other.aliveCells != null) | |
return false; | |
} else if (!aliveCells.equals(other.aliveCells)) | |
return false; | |
return true; | |
} | |
} |
Following the idea in the Clojure version I managed to simplify the Rules interface a lot:
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 inNextGeneration( | |
boolean alive, | |
int numberOfNeighbors); | |
} |
This new code keeps the functionality and characteristics of the previous version but it's much simpler.
No comments:
Post a Comment