"...finding and expressing symmetry is a preliminary step to removing duplication. If a similar thought exists in several places in the code, making them symmetrical to each other is a first good step towards unifying them"In this post we'll look at an example of expressing symmetry as a way to make duplication more visible. This is the initial code of a version of a subset of the Mars Rover kata that Álvaro García and I used in a refactoring workshop some time ago:
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
public class Rover { | |
private String direction; | |
private int y; | |
private int x; | |
public Rover(int x, int y, String direction) { | |
this.direction = direction; | |
this.y = y; | |
this.x = x; | |
} | |
public void receive(String commandsSequence) { | |
for (int i = 0; i < commandsSequence.length(); ++i) { | |
String command = commandsSequence.substring(i, i + 1); | |
if (command.equals("l") || command.equals("r")) { | |
// Rotate Rover | |
if (direction.equals("N")) { | |
if (command.equals("r")) { | |
direction = "E"; | |
} else { | |
direction = "W"; | |
} | |
} else if (direction.equals("S")) { | |
if (command.equals("r")) { | |
direction = "W"; | |
} else { | |
direction = "E"; | |
} | |
} else if (direction.equals("W")) { | |
if (command.equals("r")) { | |
direction = "N"; | |
} else { | |
direction = "S"; | |
} | |
} else { | |
if (command.equals("r")) { | |
direction = "S"; | |
} else { | |
direction = "N"; | |
} | |
} | |
} else { | |
// Displace Rover | |
int displacement1 = -1; | |
if (command.equals("f")) { | |
displacement1 = 1; | |
} | |
int displacement = displacement1; | |
if (direction.equals("N")) { | |
y += displacement; | |
} else if (direction.equals("S")) { | |
y -= displacement; | |
} else if (direction.equals("W")) { | |
x -= displacement; | |
} else { | |
x += displacement; | |
} | |
} | |
} | |
} | |
} |
- Which command to execute depending on a command code.
- How to execute the command depending on the direction the rover faces.
If we start applying Replace Type Code with State/Strategy refactoring to the code as it is now, there is a subtle problem, though.
Looking carefully more carefully, we can observe that in the case of the rover's displacement, the sequence of two decisions is clearly there:
However, that sequence is not there in the case of the rover's rotations. In this case there's a third decision (conditional) based on the command type which as, we'll see, is not necessary:
This difference between the two sequences of decisions reveals a lack of symmetry in the code. It's important to remove it before starting to refactor the code to substitute the conditionals with polymorphism.
I've seen many cases in which developers naively start extracting methods and substituting the conditionals with polymorphism starting from asymmetrical code like this one. This often leads them to create entangled and leaking abstractions.
Particularly, in this case, blindly applying Replace Type Code with State/Strategy refactoring to the left and right rotations can very likely lead to a solution containing code like the following:
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
public enum Direction { | |
private final String value; | |
Direction(String value) { | |
this.value = value; | |
} | |
public static Direction from(String value) { | |
for (Direction current : values()) { | |
if (current.value.equals(value)) { | |
return current; | |
} | |
} | |
throw new RuntimeException("Passed value is not correct: " + value); | |
} | |
public Direction rotate(Command command) { | |
if (Command.RIGHT == command) { | |
return rotateRight(); | |
} else { | |
return rotateLeft(); | |
} | |
} | |
protected abstract Direction rotateRight(); | |
protected abstract Direction rotateLeft(); | |
NORTH("N") { | |
@Override | |
protected Direction rotateRight() { | |
return EAST; | |
} | |
@Override | |
protected Direction rotateLeft() { | |
return WEST; | |
} | |
}, | |
SOUTH("S") { | |
@Override | |
protected Direction rotateRight() { | |
return (WEST); | |
} | |
@Override | |
protected Direction rotateLeft() { | |
return (EAST); | |
} | |
}, | |
WEST("W") { | |
@Override | |
protected Direction rotateRight() { | |
return (NORTH); | |
} | |
@Override | |
protected Direction rotateLeft() { | |
return (SOUTH); | |
} | |
}, | |
EAST("E") { | |
@Override | |
protected Direction rotateRight() { | |
return (SOUTH); | |
} | |
@Override | |
protected Direction rotateLeft() { | |
return (NORTH); | |
} | |
}; | |
} |
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
public enum Command { | |
RIGHT("r"), | |
LEFT("l"); | |
private String value; | |
Command(String value) { | |
this.value = value; | |
} | |
public static Command from(String value) { | |
for (Command current : values()) { | |
if (current.value.equals(value)) { | |
return current; | |
} | |
} | |
throw new RuntimeException("Passed value is not correct: " + value); | |
} | |
} |
This is bad... Some knowledge about the commands encoding has leaked from the Command class into the Direction class. Direction shouldn't be taking that decision in the first place. Neither should it know how commands are encoded. Moreover, this is a decision that was already taken and doesn't need to be taken again.
How can we avoid this trap?
We should go back to the initial code and instead of mechanically applying refactoring, we should start by removing the asymmetry from the initial code.
You can see one way of doing it in this video:
After this refactoring all cases are symmetrical (they present the same sequence of decisions)
We've not only removed the asymmetry but we've also made more explicit the case statements (Case Statements smell) on the command encoding and the duplicated switches on the direction the rover is facing:
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
public void receive(String commandsSequence) { | |
for (int i = 0; i < commandsSequence.length(); ++i) { | |
String command = commandsSequence.substring(i, i + 1); | |
if (command.equals("l")) { | |
// Rotate Rover to the left | |
if (direction.equals("N")) { | |
direction = "W"; | |
} else if (direction.equals("S")) { | |
direction = "E"; | |
} else if (direction.equals("W")) { | |
direction = "S"; | |
} else { | |
direction = "N"; | |
} | |
} else if (command.equals("r")) { | |
// Rotate Rover to the right | |
if (direction.equals("N")) { | |
direction = "E"; | |
} else if (direction.equals("S")) { | |
direction = "W"; | |
} else if (direction.equals("W")) { | |
direction = "N"; | |
} else { | |
direction = "S"; | |
} | |
} else { | |
// Displace Rover | |
int displacement1 = -1; | |
if (command.equals("f")) { | |
displacement1 = 1; | |
} | |
int displacement = displacement1; | |
if (direction.equals("N")) { | |
y += displacement; | |
} else if (direction.equals("S")) { | |
y -= displacement; | |
} else if (direction.equals("W")) { | |
x -= displacement; | |
} else { | |
x += displacement; | |
} | |
} | |
} | |
} |
If we start the Replace Type Code with State/Strategy refactoring now, it's more likely that we'll end with a code in which Direction knows nothing about how the commands are encoded:
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
public enum Direction { | |
NORTH { | |
@Override | |
public Coordinates move(Coordinates coordinates, int displacement) { | |
return coordinates.incrementY(displacement); | |
} | |
@Override | |
public Direction rotateRight() { | |
return EAST; | |
} | |
@Override | |
public Direction rotateLeft() { | |
return WEST; | |
} | |
}, SOUTH { | |
@Override | |
public Coordinates move(Coordinates coordinates, int displacement) { | |
return coordinates.incrementY(-displacement); | |
} | |
@Override | |
public Direction rotateRight() { | |
return WEST; | |
} | |
@Override | |
public Direction rotateLeft() { | |
return EAST; | |
} | |
}, EAST { | |
@Override | |
public Coordinates move(Coordinates coordinates, int displacement) { | |
return coordinates.incrementX(displacement); | |
} | |
@Override | |
public Direction rotateRight() { | |
return SOUTH; | |
} | |
@Override | |
public Direction rotateLeft() { | |
return NORTH; | |
} | |
}, WEST { | |
@Override | |
public Coordinates move(Coordinates coordinates, int displacement) { | |
return coordinates.incrementX(-displacement); | |
} | |
@Override | |
public Direction rotateRight() { | |
return NORTH; | |
} | |
public Direction rotateLeft() { | |
return SOUTH; | |
} | |
}; | |
public static Direction pointingTo(String directionCode) { | |
if (directionCode.equals("N")) { | |
return NORTH; | |
} else if (directionCode.equals("S")) { | |
return SOUTH; | |
} else if (directionCode.equals("E")) { | |
return EAST; | |
} else { | |
return WEST; | |
} | |
} | |
abstract public Coordinates move(Coordinates coordinates, int displacement); | |
abstract public Direction rotateRight(); | |
abstract public Direction rotateLeft(); | |
} |
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
public class CommandCodeInterpreter { | |
private static final int DISPLACEMENT_LENGTH = 1; | |
private static final String FORWARD_MOVEMENT = "f"; | |
private static final String BACKWARD_MOVEMENT = "b"; | |
private static final String RIGHT_ROTATION = "r"; | |
private static final String LEFT_ROTATION = "l"; | |
private static Map<String, Command> knownCommands = knownCommands(); | |
public static Command interpret(String commandCode) { | |
return getCommandFor(commandCode); | |
} | |
private static Command getCommandFor(String commandCode) { | |
return knownCommands.get(commandCode); | |
} | |
private static Map<String, Command> knownCommands() { | |
Map<String, Command> knownCommands = new HashMap<>(); | |
knownCommands.put(FORWARD_MOVEMENT, new Movement(DISPLACEMENT_LENGTH)); | |
knownCommands.put(BACKWARD_MOVEMENT, new Movement(-DISPLACEMENT_LENGTH)); | |
knownCommands.put(RIGHT_ROTATION, new RightRotation()); | |
knownCommands.put(LEFT_ROTATION, new LeftRotation()); | |
return Collections.unmodifiableMap(knownCommands); | |
} | |
} |
Then, by removing those asymmetries, you can make the duplication more visible and disentangle the entangled dimensions of complexity. The only thing you need is to be patient and don't start "obvious" refactorings before thinking a bit about symmetry.
[1] Dimension of Complexity is a term used by Mateu Adsuara in a talk at SocraCan16 to name an orthogonal functionality. In that talk he used dimensions of complexity to group the examples in his test list and help him choose the next test when doing TDD. He talked about it in this three posts: Complexity dimensions - FizzBuzz part I, Complexity dimensions - FizzBuzz part II and Complexity dimensions - FizzBuzz part III. Other names for the same concept that I've heard are axes of change, directions of change or vectors of change.
No comments:
Post a Comment