In this version I've used protocols instead of multimethods.
Since I had previously distributed the code in several different name spaces, I only had to modify the rover name space.
The rest of the code keeps just using the same functions of the rover name space: the rover factory and the four commands rotate-left, rotate-right, move-forwards and move-backwards.
However, the implementation of the rover name space is quite different now.
This is the new code:
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
(ns mars-rover.rover) | |
(defn rover [x y direction] | |
{:x x :y y :direction direction}) | |
(defprotocol Rotable | |
(rotate-left [this]) | |
(rotate-right [this])) | |
(defprotocol Movable | |
(move-forwards [this]) | |
(move-backwards [this])) | |
(defrecord FacingNorth [x y] | |
Rotable | |
(rotate-left | |
[{:keys [x y]}] | |
(rover x y :west)) | |
(rotate-right | |
[{:keys [x y]}] | |
(rover x y :east)) | |
Movable | |
(move-forwards | |
[{:keys [x y]}] | |
(rover x (inc y) :north)) | |
(move-backwards | |
[{:keys [x y]}] | |
(rover x (dec y) :north))) | |
(defrecord FacingSouth [x y] | |
Rotable | |
(rotate-left | |
[{:keys [x y]}] | |
(rover x y :east)) | |
(rotate-right | |
[{:keys [x y]}] | |
(rover x y :west)) | |
Movable | |
(move-forwards | |
[{:keys [x y]}] | |
(rover x (dec y) :south)) | |
(move-backwards | |
[{:keys [x y]}] | |
(rover x (inc y) :south))) | |
(defrecord FacingEast [x y] | |
Rotable | |
(rotate-left | |
[{:keys [x y]}] | |
(rover x y :north)) | |
(rotate-right | |
[{:keys [x y]}] | |
(rover x y :south)) | |
Movable | |
(move-forwards | |
[{:keys [x y]}] | |
(rover (inc x) y :east)) | |
(move-backwards | |
[{:keys [x y]}] | |
(rover (dec x) y :east))) | |
(defrecord FacingWest [x y] | |
Rotable | |
(rotate-left | |
[{:keys [x y]}] | |
(rover x y :south)) | |
(rotate-right | |
[{:keys [x y]}] | |
(rover x y :north)) | |
Movable | |
(move-forwards | |
[{:keys [x y]}] | |
(rover (dec x) y :west)) | |
(move-backwards | |
[{:keys [x y]}] | |
(rover (inc x) y :west))) | |
(defn oriented-rover [{:keys [x y direction]}] | |
(case direction | |
:north (FacingNorth. x y) | |
:south (FacingSouth. x y) | |
:east (FacingEast. x y) | |
:west (FacingWest. x y))) | |
(defn rotate-left [rover] | |
(rotate-left (oriented-rover rover))) | |
(defn rotate-right [rover] | |
(rotate-right (oriented-rover rover))) | |
(defn move-forwards [rover] | |
(move-forwards (oriented-rover rover))) | |
(defn move-backwards [rover] | |
(move-backwards (oriented-rover rover))) |
Depending on the value of the rover direction, the oriented-rover factory function, creates an instance of one of those records.
When one of the command functions (rotate-left, rotate-right, move-forwards or move-backwards) is called, it calls a function with the same name passing it the result of calling the oriented-rover factory on the rover.
This call is dispatched on the type of the record the call to oriented-rover returns, so that, in the end, the right implementation of the function gets called.
This has been a nice exercise to explore and practice with protocols.
I'll continue exploring other ways to solve this kata with Clojure.
You can find the code in this GitHub repository.
-----------
Update: I continued working in this code on Mars Rover using a finite state machine implemented with mutually recursive functions and trampoline
No comments:
Post a Comment