The first big change is that this version does not make use of any global variables. A key concept of functional programming is that calling a function does not have any side effects. Any time we modify a value, such as attacking the player and decreasing his health, we cannot modify any external state. Rather we pass in the player as a parameter to the function, and generate a new object with this modified state and use the returned value later on in the program. Since we no longer have the global variables keeping track of the players stats, we need to create a structure to keep track of this:
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
(defn create-player [max-health max-agility max-strength] | |
{:health max-health :max-health max-health | |
:agility max-agility :max-agility max-agility | |
:strength max-strength :max-strength max-strength}) |
For our first function, lets create one that damages a player's attribute by some amount. So given the player, attribute and damage we want to return a new player structure with the given attribute altered. We can use the assoc function, which takes a hash-map, the key and the new value. This function will return a new hash-map where everything is the same except for the specified key, which now has the new value.
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
(defn hit-player [player attr dmg] | |
(assoc player attr (max 0 (- (attr player) dmg)))) | |
orc.core> (def player (create-player 20 20 20)) | |
#'orc.core/player | |
orc.core> (hit-player player :health 5) | |
{:health 15, :max-health 20, :agility 20, :max-agility 20, :strength 20, :max-strength 20} | |
orc.core> player | |
{:health 20, :max-health 20, :agility 20, :max-agility 20, :strength 20, :max-strength 20} |
The next piece is dealing with the monster methods and dispatching on type. In clojure there are several approaches to dealing with inheritance. One approach uses defrecord and protocols. I tried that way first, but with little experience using it, I found the resulting code less clear. An alternate approach that I used was multi methods. The main compents of a multimethod are the method name and the dispatch function. Since a key (e.g. :some-key) can act as a function, that means you can dispatch the method based on the value of a given key or set of keys of a map object. So in this case we have the following method signatures and implementations:
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
(defmulti monster-show :type) | |
(defmulti monster-attack :type) | |
(defmulti monster-hit :type) | |
::monster | |
(derive ::orc ::monster) | |
(derive ::hydra ::monster) | |
(defn orc [health club-level] | |
{:type ::orc :health health :club-level club-level}) | |
(defn hydra [health] | |
{:type ::hydra :health health}) | |
(defmethod monster-show ::monster [m] | |
(str "A " (monster-name (:type m)) " with " (:health m) " health.")) | |
(defmethod monster-show ::hydra [m] | |
(str "A hydra with " (:health m) " heads!")) |
We also need some addition methods to help with I/O. When using swank in emacs, the regular read-line method does not work. Fortunately swank-clojure provides a method to work around this problem. Also, read-line returns a string with a newline character so we need a method to parse the integer.
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
(defn read-from-in [] | |
(swank.core/with-read-line-support (read-line))) | |
(defn read-int [] | |
(let [x (clojure.string/replace (read-from-in) "\n" "")] | |
(if (every? #(Character/isDigit %) x) | |
(Integer/parseInt x) | |
-1))) |
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
(defn monster-round-original [player monsters] | |
(loop [n 0 p player] | |
(if (>= n (count monsters)) | |
p | |
(recur (inc n) | |
(if (monster-dead? (nth monsters n)) | |
p | |
(let [r (monster-attack (nth monsters n) p)] | |
(print (:attack r)) | |
(:player r))))))) | |
(defn- attack-player [player monster] | |
(if (monster-dead? monster) | |
player | |
(let [r (monster-attack monster player)] | |
(print (:attack r)) | |
(:player r)))) | |
(defn monster-round-improved [player monsters] | |
(reduce attack-player player monsters)) | |
(defn- attack-monsters [monsters player] | |
(if (monsters-dead? monsters) | |
monsters | |
(do | |
(show-monsters monsters) | |
(player-attack player monsters)))) | |
(defn num-attacks [player] | |
(inc (Math/floor (/ (max 0 (:agility player)) 15)))) | |
(defn player-round [player monsters] | |
(reduce attack-monsters monsters | |
(repeat (num-attacks player) player))) |
The one place where I did stick with a loop/recur was with the main game loop. Since I am not specifically using a data structure to loop on, I am uncertain if there is a better way to implement this.
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
(defn level-loop [player monsters] | |
(loop [m monsters p player] | |
(if (or (monsters-dead? m) (player-dead? p)) | |
p | |
(do | |
(print (str "\n" (show-player p))) | |
(let [m2 (player-round p m)] | |
(recur m2 (monster-round p m2))))))) |
The full code can be found here along with the original common lisp code.