My partner Fernando Mora and I used TDD in Scala to write the code.
Today I did it once again in Clojure.
I'd like to explain here how I did it step by step in order to share it with the Barcelona Software Craftsmanship members.
I started by writing this test:
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 eratosthenes-sieve.core-test | |
(:use midje.sweet) | |
(:use [eratosthenes-sieve.core])) | |
(facts | |
"about Eratosthenes sieve" | |
(fact | |
"it returns all the primes up to a given number" | |
(primes-up-to 2) => [2])) |
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 eratosthenes-sieve.core) | |
(defn primes-up-to [n] | |
[2]) |
Then I wrote the following test:
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 eratosthenes-sieve.core-test | |
(:use midje.sweet) | |
(:use [eratosthenes-sieve.core])) | |
(facts | |
"about Eratosthenes sieve" | |
(fact | |
"it returns all the primes up to a given number" | |
(primes-up-to 2) => [2] | |
(primes-up-to 3) => [2 3])) |
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 eratosthenes-sieve.core) | |
(defn primes-up-to [n] | |
(range 2 (inc n))) |
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 eratosthenes-sieve.core-test | |
(:use midje.sweet) | |
(:use [eratosthenes-sieve.core])) | |
(facts | |
"about Eratosthenes sieve" | |
(fact | |
"it returns all the primes up to a given number" | |
(primes-up-to 2) => [2] | |
(primes-up-to 3) => [2 3] | |
(primes-up-to 5) => [2 3 5])) |
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 eratosthenes-sieve.core) | |
(defn primes-up-to [n] | |
(filter #(or (not= 0 (mod % 2)) (= % 2)) | |
(range 2 (inc n)))) |
Since the implementation was obvious and I could rely on filter, I decided not to follow that path and took a larger step.
I've noticed that my TDD baby steps tend to be larger in functional languages. I think the reason is, on one hand, the REPL which complements TDD by providing a faster feedback loop for triangulation and trying things out and, on the other hand, the power of sequence functions in those languages (in the case of this kata the Scala and Clojure ones).
Once the test was passing I started to refactor that ugly one-liner and got to this more readable version:
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 eratosthenes-sieve.core) | |
(defn- integers-up-to [n] | |
(range 2 (inc n))) | |
(defn- multiple-of? [n p] | |
(and (zero? (mod n p)) (not= n p))) | |
(defn primes-up-to [n] | |
(remove #(multiple-of? % 2) (integers-up-to n))) |
My next goal was to write a test to drive me to generalize the code a bit more by eliminating the hard-coded number 2 in line 10 of the code.
To do it I just needed a test that forced the code to also eliminate just the multiples of 3:
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 eratosthenes-sieve.core-test | |
(:use midje.sweet) | |
(:use [eratosthenes-sieve.core])) | |
(facts | |
"about Eratosthenes sieve" | |
(fact | |
"it returns all the primes up to a given number" | |
(primes-up-to 2) => [2] | |
(primes-up-to 3) => [2 3] | |
(primes-up-to 5) => [2 3 5] | |
(primes-up-to 11) => [2 3 5 7 11])) |
But before doing that, I quickly went to green by calling the sieve function twice to eliminate the multiples of 2 and 3:
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 eratosthenes-sieve.core) | |
(defn- integers-up-to [n] | |
(range 2 (inc n))) | |
(defn- multiple-of? [n p] | |
(and (zero? (mod n p)) (not= n p))) | |
(defn- sieve [numbers prime] | |
(remove #(multiple-of? % prime) numbers)) | |
(defn primes-up-to [n] | |
(sieve (sieve (integers-up-to n) 2) 3)) |
In this case I think that having tried to directly introduce recursion to make the test pass would have been too large a step to take, so I played safe.
Once in green again I safely refactored the code to introduce recursion:
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 eratosthenes-sieve.core) | |
(defn- integers-up-to [n] | |
(range 2 (inc n))) | |
(defn- multiple-of? [n p] | |
(and (zero? (mod n p)) (not= n p))) | |
(defn- sieve [numbers prime] | |
(let [sieved (remove #(multiple-of? % prime) numbers) | |
next-prime (first (drop-while #(<= % prime) sieved))] | |
(if (nil? next-prime) | |
sieved | |
(recur sieved next-prime)))) | |
(defn primes-up-to [n] | |
(sieve (integers-up-to n) 2)) |
From this point on I just refactored the code trying to make it a bit more readable until I got to this version:
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 eratosthenes-sieve.core) | |
(defn- integers-up-to [n] | |
(range 2 (inc n))) | |
(defn- multiple-of? [n p] | |
(and (zero? (mod n p)) (not= n p))) | |
(defn- next-prime [prime numbers] | |
(first (drop-while #(<= % prime) numbers))) | |
(defn- sieve-multiples-of [prime numbers] | |
(remove #(multiple-of? % prime) numbers)) | |
(defn- sieve [numbers] | |
(loop [primes numbers prime 2] | |
(let [sieved (sieve-multiples-of prime primes)] | |
(if-let [prime (next-prime prime sieved)] | |
(recur sieved prime) | |
primes)))) | |
(defn primes-up-to [n] | |
(sieve (integers-up-to n))) |
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 eratosthenes-sieve.core-test | |
(:use midje.sweet) | |
(:use [eratosthenes-sieve.core])) | |
(facts | |
"about Eratosthenes sieve" | |
(fact | |
"it returns all the primes up to a given number" | |
(primes-up-to 100) => [2 3 5 7 11 13 17 19 23 29 31 37 41 43 47 53 59 61 67 71 73 79 83 89 97])) |
That's all.
I'd like to thank Barcelona Software Craftsmanship members for practicing together every two weeks, especially Álvaro García for facilitating this last kata, and eBay España for kindly having us yesterday (and on many previous events) in their Barcelona office.
Nice play by play TDD!!
ReplyDeleteIn the refactored version I thought that the p in sieve meant prime, but it doesn't. I like better the version of sieve on step 6. May be sieve could only remove the prime given and have another function controlling next prime / sieve and recursion which will only take the (range (inc n))
Thanks a lot for sharing!! Reading this post was like being there while you solved the kata.
--
Francesc
I followed your advice.
ReplyDeleteThanks Francesc!
Best regards,
M