Reformat all koan files for REPL compatibility.

This commit is contained in:
frizzbolt 2016-02-26 20:33:05 -06:00
parent e1cc8a72b4
commit dc460acdda
22 changed files with 224 additions and 357 deletions

View file

@ -1,70 +1,66 @@
(ns koans.02-strings
(:require [koan-engine.core :refer :all]
[clojure.string :as string]))
(meditations ; "A string is nothing more than text surrounded by double quotes"
"A string is nothing more than text surrounded by double quotes"
(= __ "hello") (= __ "hello")
"But double quotes are just magic on top of something deeper" ; "But double quotes are just magic on top of something deeper"
(= __ (str 'world)) (= __ (str 'world))
"You can do more than create strings, you can put them together" ; "You can do more than create strings, you can put them together"
(= "Cool right?" (str __ __)) (= "Cool right?" (str __ __))
"You can even get certain characters" ; "You can even get certain characters"
(= \C (get "Characters" __)) (= \C (get "Characters" __))
"Or even count the characters" ; "Or even count the characters"
(= __ (count "Hello World")) (= __ (count "Hello World"))
"But strings and characters are not the same" ; "But strings and characters are not the same"
(= __ (= \c "c")) (= __ (= \c "c"))
"What if you only wanted to get part of a string?" ; "What if you only wanted to get part of a string?"
(= "World" (subs "Hello World" __ __)) (= "World" (subs "Hello World" __ __))
"How about joining together elements in a list?" ; "How about joining together elements in a list?"
(= __ (string/join '(1 2 3))) (= __ (string/join '(1 2 3)))
"What if you wanted to separate them out?" ; "What if you wanted to separate them out?"
(= "1, 2, 3" (string/join __ '(1 2 3))) (= "1, 2, 3" (string/join __ '(1 2 3)))
"Maybe you want to separate out all your lines" ; "Maybe you want to separate out all your lines"
(= [__ __ __] (string/split-lines "1\n2\n3")) (= [__ __ __] (string/split-lines "1\n2\n3"))
"You may want to make sure your words are backwards" ; "You may want to make sure your words are backwards"
(= __ (string/reverse "hello")) (= __ (string/reverse "hello"))
"Maybe you want to find the index of the first occurence of a substring" ; "Maybe you want to find the index of the first occurence of a substring"
(= 0 (string/index-of "hello world" __)) (= 0 (string/index-of "hello world" __))
"Or maybe the last index of the same" ; "Or maybe the last index of the same"
(= __ (string/last-index-of "hello world, hello" "hello")) (= __ (string/last-index-of "hello world, hello" "hello"))
"But when something doesn't exist, nothing is found" ; "But when something doesn't exist, nothing is found"
(= __ (string/index-of "hello world" "bob")) (= __ (string/index-of "hello world" "bob"))
"Sometimes you don't want whitespace cluttering the front and back" ; "Sometimes you don't want whitespace cluttering the front and back"
(= __ (string/trim " \nhello world \t \n")) (= __ (string/trim " \nhello world \t \n"))
"You can check if something is a char" ; "You can check if something is a char"
(= __ (char? \c)) (= __ (char? \c))
"But it may not be" ; "But it may not be"
(= __ (char? "a")) (= __ (char? "a"))
"But chars aren't strings" ; "But chars aren't strings"
(= __ (string? \b)) (= __ (string? \b))
"Strings are strings" ; "Strings are strings"
(= true (string? __)) (= true (string? __))
"Some strings may be blank" ; "Some strings may be blank"
(= __ (string/blank? "")) (= __ (string/blank? ""))
"Even if at first glance they aren't" ; "Even if at first glance they aren't"
(= __ (string/blank? " \n \t ")) (= __ (string/blank? " \n \t "))
"However, most strings aren't blank" ; "However, most strings aren't blank"
(= __ (string/blank? "hello?\nare you out there?"))) (= __ (string/blank? "hello?\nare you out there?"))

View file

@ -1,45 +1,42 @@
(ns koans.03-lists
(:require [koan-engine.core :refer :all]))
(meditations ; "Lists can be expressed by function or a quoted form"
"Lists can be expressed by function or a quoted form"
(= '(__ __ __ __ __) (list 1 2 3 4 5)) (= '(__ __ __ __ __) (list 1 2 3 4 5))
"They are Clojure seqs (sequences), so they allow access to the first" ; "They are Clojure seqs (sequences), so they allow access to the first"
(= __ (first '(1 2 3 4 5))) (= __ (first '(1 2 3 4 5)))
"As well as the rest" ; "As well as the rest"
(= __ (rest '(1 2 3 4 5))) (= __ (rest '(1 2 3 4 5)))
"Count your blessings" ; "Count your blessings"
(= __ (count '(dracula dooku chocula))) (= __ (count '(dracula dooku chocula)))
"Before they are gone" ; "Before they are gone"
(= __ (count '())) (= __ (count '()))
"The rest, when nothing is left, is empty" ; "The rest, when nothing is left, is empty"
(= __ (rest '(100))) (= __ (rest '(100)))
"Construction by adding an element to the front is easy" ; "Construction by adding an element to the front is easy"
(= __ (cons :a '(:b :c :d :e))) (= __ (cons :a '(:b :c :d :e)))
"Conjoining an element to a list isn't hard either" ; "Conjoining an element to a list isn't hard either"
(= __ (conj '(:a :b :c :d) :e)) (= __ (conj '(:a :b :c :d) :e))
"You can use a list like a stack to get the first element" ; "You can use a list like a stack to get the first element"
(= __ (peek '(:a :b :c :d :e))) (= __ (peek '(:a :b :c :d :e)))
"Or the others" ; "Or the others"
(= __ (pop '(:a :b :c :d :e))) (= __ (pop '(:a :b :c :d :e)))
"But watch out if you try to pop nothing" ; "But watch out if you try to pop nothing"
(= __ (try (= __ (try
(pop '()) (pop '())
(catch IllegalStateException e (catch IllegalStateException e
"No dice!"))) "No dice!")))
"The rest of nothing isn't so strict" ; "The rest of nothing isn't so strict"
(= __ (try (= __ (try
(rest '()) (rest '())
(catch IllegalStateException e (catch IllegalStateException e
"No dice!")))) "No dice!")))

View file

@ -1,33 +1,30 @@
(ns koans.04-vectors
(:require [koan-engine.core :refer :all]))
(meditations ; "You can use vectors in clojure as array-like structures"
"You can use vectors in clojure as array-like structures"
(= __ (count [42])) (= __ (count [42]))
"You can create a vector from a list" ; "You can create a vector from a list"
(= __ (vec '(1))) (= __ (vec '(1)))
"Or from some elements" ; "Or from some elements"
(= __ (vector nil nil)) (= __ (vector nil nil))
"But you can populate it with any number of elements at once" ; "But you can populate it with any number of elements at once"
(= [1 __] (vec '(1 2))) (= [1 __] (vec '(1 2)))
"Conjoining to a vector is different than to a list" ; "Conjoining to a vector is different than to a list"
(= __ (conj [111 222] 333)) (= __ (conj [111 222] 333))
"You can get the first element of a vector like so" ; "You can get the first element of a vector like so"
(= __ (first [:peanut :butter :and :jelly])) (= __ (first [:peanut :butter :and :jelly]))
"And the last in a similar fashion" ; "And the last in a similar fashion"
(= __ (last [:peanut :butter :and :jelly])) (= __ (last [:peanut :butter :and :jelly]))
"Or any index if you wish" ; "Or any index if you wish"
(= __ (nth [:peanut :butter :and :jelly] 3)) (= __ (nth [:peanut :butter :and :jelly] 3))
"You can also slice a vector" ; "You can also slice a vector"
(= __ (subvec [:peanut :butter :and :jelly] 1 3)) (= __ (subvec [:peanut :butter :and :jelly] 1 3))
"Equality with collections is in terms of values" ; "Equality with collections is in terms of values"
(= (list 1 2 3) (vector 1 2 __))) (= (list 1 2 3) (vector 1 2 __))

View file

@ -1,22 +1,18 @@
(ns koans.05-sets
(:require [koan-engine.core :refer :all]
[clojure.set :as set]))
(meditations ; "You can create a set by converting another collection"
"You can create a set by converting another collection"
(= #{3} (set __)) (= #{3} (set __))
"Counting them is like counting other collections" ; "Counting them is like counting other collections"
(= __ (count #{1 2 3})) (= __ (count #{1 2 3}))
"Remember that a set is a *mathematical* set" ; "Remember that a set is a *mathematical* set"
(= __ (set '(1 1 2 2 3 3 4 4 5 5))) (= __ (set '(1 1 2 2 3 3 4 4 5 5)))
"You can ask clojure for the union of two sets" ; "You can ask clojure for the union of two sets"
(= __ (set/union #{1 2 3 4} #{2 3 5})) (= __ (set/union #{1 2 3 4} #{2 3 5}))
"And also the intersection" ; "And also the intersection"
(= __ (set/intersection #{1 2 3 4} #{2 3 5})) (= __ (set/intersection #{1 2 3 4} #{2 3 5}))
"But don't forget about the difference" ; "But don't forget about the difference"
(= __ (set/difference #{1 2 3 4 5} #{2 3 5}))) (= __ (set/difference #{1 2 3 4 5} #{2 3 5}))

View file

@ -1,50 +1,47 @@
(ns koans.06-maps
(:require [koan-engine.core :refer :all]))
(meditations ; "Don't get lost when creating a map"
"Don't get lost when creating a map"
(= {:a 1 :b 2} (hash-map :a 1 __ __)) (= {:a 1 :b 2} (hash-map :a 1 __ __))
"A value must be supplied for each key" ; "A value must be supplied for each key"
(= {:a 1} (hash-map :a __)) (= {:a 1} (hash-map :a __))
"The size is the number of entries" ; "The size is the number of entries"
(= __ (count {:a 1 :b 2})) (= __ (count {:a 1 :b 2}))
"You can look up the value for a given key" ; "You can look up the value for a given key"
(= __ (get {:a 1 :b 2} :b)) (= __ (get {:a 1 :b 2} :b))
"Maps can be used as functions to do lookups" ; "Maps can be used as functions to do lookups"
(= __ ({:a 1 :b 2} :a)) (= __ ({:a 1 :b 2} :a))
"And so can keywords" ; "And so can keywords"
(= __ (:a {:a 1 :b 2})) (= __ (:a {:a 1 :b 2}))
"But map keys need not be keywords" ; "But map keys need not be keywords"
(= __ ({2010 "Vancouver" 2014 "Sochi" 2018 "PyeongChang"} 2014)) (= __ ({2010 "Vancouver" 2014 "Sochi" 2018 "PyeongChang"} 2014))
"You may not be able to find an entry for a key" ; "You may not be able to find an entry for a key"
(= __ (get {:a 1 :b 2} :c)) (= __ (get {:a 1 :b 2} :c))
"But you can provide your own default" ; "But you can provide your own default"
(= __ (get {:a 1 :b 2} :c :key-not-found)) (= __ (get {:a 1 :b 2} :c :key-not-found))
"You can find out if a key is present" ; "You can find out if a key is present"
(= __ (contains? {:a nil :b nil} :b)) (= __ (contains? {:a nil :b nil} :b))
"Or if it is missing" ; "Or if it is missing"
(= __ (contains? {:a nil :b nil} :c)) (= __ (contains? {:a nil :b nil} :c))
"Maps are immutable, but you can create a new and improved version" ; "Maps are immutable, but you can create a new and improved version"
(= {1 "January" 2 __} (assoc {1 "January"} 2 "February")) (= {1 "January" 2 __} (assoc {1 "January"} 2 "February"))
"You can also create a new version with an entry removed" ; "You can also create a new version with an entry removed"
(= {__ __} (dissoc {1 "January" 2 "February"} 2)) (= {__ __} (dissoc {1 "January" 2 "February"} 2))
"Often you will need to get the keys, but the order is undependable" ; "Often you will need to get the keys, but the order is undependable"
(= (list __ __ __) (= (list __ __ __)
(sort (keys { 2014 "Sochi" 2018 "PyeongChang" 2010 "Vancouver"}))) (sort (keys { 2014 "Sochi" 2018 "PyeongChang" 2010 "Vancouver"})))
"You can get the values in a similar way" ; "You can get the values in a similar way"
(= (list __ __ __) (= (list __ __ __)
(sort (vals {2010 "Vancouver" 2014 "Sochi" 2018 "PyeongChang"})))) (sort (vals {2010 "Vancouver" 2014 "Sochi" 2018 "PyeongChang"})))

View file

@ -1,40 +1,32 @@
(ns koans.07-functions
(:require [koan-engine.core :refer :all]))
(defn multiply-by-ten [n] ; "Calling a function is like giving it a hug with parentheses"
(* 10 n))
(defn square [n] (* n n))
(meditations
"Calling a function is like giving it a hug with parentheses"
(= __ (square 9)) (= __ (square 9))
"Functions are usually defined before they are used" ; "Functions are usually defined before they are used"
(= __ (multiply-by-ten 2)) (= __ (multiply-by-ten 2))
"But they can also be defined inline" ; "But they can also be defined inline"
(= __ ((fn [n] (* 5 n)) 2)) (= __ ((fn [n] (* 5 n)) 2))
"Or using an even shorter syntax" ; "Or using an even shorter syntax"
(= __ (#(* 15 %) 4)) (= __ (#(* 15 %) 4))
"Even anonymous functions may take multiple arguments" ; "Even anonymous functions may take multiple arguments"
(= __ (#(+ %1 %2 %3) 4 5 6)) (= __ (#(+ %1 %2 %3) 4 5 6))
"Arguments can also be skipped" ; "Arguments can also be skipped"
(= __ (#(* 15 %2) 1 2)) (= __ (#(* 15 %2) 1 2))
"One function can beget another" ; "One function can beget another"
(= 9 (((fn [] ___)) 4 5)) (= 9 (((fn [] ___)) 4 5))
"Functions can also take other functions as input" ; "Functions can also take other functions as input"
(= 20 ((fn [f] (f 4 5)) (= 20 ((fn [f] (f 4 5))
___)) ___))
"Higher-order functions take function arguments" ; "Higher-order functions take function arguments"
(= 25 (___ (= 25 (___
(fn [n] (* n n)))) (fn [n] (* n n))))
"But they are often better written using the names of functions" ; "But they are often better written using the names of functions"
(= 25 (___ square))) (= 25 (___ square))

View file

@ -1,47 +1,37 @@
(ns koans.08-conditionals
(:require [koan-engine.core :refer :all]))
(defn explain-exercise-velocity [exercise-term] ; "You will face many decisions"
(case exercise-term
:bicycling "pretty fast"
:jogging "not super fast"
:walking "not fast at all"
"is that even exercise?"))
(meditations
"You will face many decisions"
(= __ (if (false? (= 4 5)) (= __ (if (false? (= 4 5))
:a :a
:b)) :b))
"Some of them leave you no alternative" ; "Some of them leave you no alternative"
(= __ (if (> 4 3) (= __ (if (> 4 3)
[])) []))
"And in such a situation you may have nothing" ; "And in such a situation you may have nothing"
(= __ (if (nil? 0) (= __ (if (nil? 0)
[:a :b :c])) [:a :b :c]))
"In others your alternative may be interesting" ; "In others your alternative may be interesting"
(= :glory (if (not (empty? ())) (= :glory (if (not (empty? ()))
:doom :doom
__)) __))
"You may have a multitude of possible paths" ; "You may have a multitude of possible paths"
(let [x 5] (let [x 5]
(= :your-road (cond (= x __) :road-not-taken (= :your-road (cond (= x __) :road-not-taken
(= x __) :another-road-not-taken (= x __) :another-road-not-taken
:else __))) :else __)))
"Or your fate may be sealed" ; "Or your fate may be sealed"
(= 'doom (if-not (zero? __) (= 'doom (if-not (zero? __)
'doom 'doom
'more-doom)) 'more-doom))
"In case of emergency, go fast" ; "In case of emergency, go fast"
(= "pretty fast" (= "pretty fast"
(explain-exercise-velocity __)) (explain-exercise-velocity __))
"But admit it when you don't know what to do" ; "But admit it when you don't know what to do"
(= __ (= __
(explain-exercise-velocity :watching-tv))) (explain-exercise-velocity :watching-tv))

View file

@ -1,35 +1,32 @@
(ns koans.09-higher-order-functions
(:require [koan-engine.core :refer :all]))
(meditations ; "The map function relates a sequence to another"
"The map function relates a sequence to another"
(= [__ __ __] (map (fn [x] (* 4 x)) [1 2 3])) (= [__ __ __] (map (fn [x] (* 4 x)) [1 2 3]))
"You may create that mapping" ; "You may create that mapping"
(= [1 4 9 16 25] (map (fn [x] __) [1 2 3 4 5])) (= [1 4 9 16 25] (map (fn [x] __) [1 2 3 4 5]))
"Or use the names of existing functions" ; "Or use the names of existing functions"
(= __ (map nil? [:a :b nil :c :d])) (= __ (map nil? [:a :b nil :c :d]))
"A filter can be strong" ; "A filter can be strong"
(= __ (filter (fn [x] false) '(:anything :goes :here))) (= __ (filter (fn [x] false) '(:anything :goes :here)))
"Or very weak" ; "Or very weak"
(= __ (filter (fn [x] true) '(:anything :goes :here))) (= __ (filter (fn [x] true) '(:anything :goes :here)))
"Or somewhere in between" ; "Or somewhere in between"
(= [10 20 30] (filter (fn [x] __) [10 20 30 40 50 60 70 80])) (= [10 20 30] (filter (fn [x] __) [10 20 30 40 50 60 70 80]))
"Maps and filters may be combined" ; "Maps and filters may be combined"
(= [10 20 30] (map (fn [x] __) (filter (fn [x] __) [1 2 3 4 5 6 7 8]))) (= [10 20 30] (map (fn [x] __) (filter (fn [x] __) [1 2 3 4 5 6 7 8])))
"Reducing can increase the result" ; "Reducing can increase the result"
(= __ (reduce (fn [a b] (* a b)) [1 2 3 4])) (= __ (reduce (fn [a b] (* a b)) [1 2 3 4]))
"You can start somewhere else" ; "You can start somewhere else"
(= 2400 (reduce (fn [a b] (* a b)) __ [1 2 3 4])) (= 2400 (reduce (fn [a b] (* a b)) __ [1 2 3 4]))
"Numbers are not the only things one can reduce" ; "Numbers are not the only things one can reduce"
(= "longest" (reduce (fn [a b] (= "longest" (reduce (fn [a b]
(if (< __ __) b a)) (if (< __ __) b a))
["which" "word" "is" "longest"]))) ["which" "word" "is" "longest"]))

View file

@ -1,42 +1,26 @@
(ns koans.10-runtime-polymorphism
(:require [koan-engine.core :refer :all]))
(defn hello ; "Some functions can be used in different ways - with no arguments"
([] "Hello World!")
([a] (str "Hello, you silly " a "."))
([a & more] (str "Hello to this group: "
(apply str
(interpose ", " (cons a more)))
"!")))
(defmulti diet (fn [x] (:eater x)))
(defmethod diet :herbivore [a] __)
(defmethod diet :carnivore [a] __)
(defmethod diet :default [a] __)
(meditations
"Some functions can be used in different ways - with no arguments"
(= __ (hello)) (= __ (hello))
"With one argument" ; "With one argument"
(= __ (hello "world")) (= __ (hello "world"))
"Or with many arguments" ; "Or with many arguments"
(= __ (= __
(hello "Peter" "Paul" "Mary")) (hello "Peter" "Paul" "Mary"))
"Multimethods allow more complex dispatching" ; "Multimethods allow more complex dispatching"
(= "Bambi eats veggies." (= "Bambi eats veggies."
(diet {:species "deer" :name "Bambi" :age 1 :eater :herbivore})) (diet {:species "deer" :name "Bambi" :age 1 :eater :herbivore}))
"Animals have different names" ; "Animals have different names"
(= "Thumper eats veggies." (= "Thumper eats veggies."
(diet {:species "rabbit" :name "Thumper" :age 1 :eater :herbivore})) (diet {:species "rabbit" :name "Thumper" :age 1 :eater :herbivore}))
"Different methods are used depending on the dispatch function result" ; "Different methods are used depending on the dispatch function result"
(= "Simba eats animals." (= "Simba eats animals."
(diet {:species "lion" :name "Simba" :age 1 :eater :carnivore})) (diet {:species "lion" :name "Simba" :age 1 :eater :carnivore}))
"You may use a default method when no others match" ; "You may use a default method when no others match"
(= "I don't know what Rich Hickey eats." (= "I don't know what Rich Hickey eats."
(diet {:name "Rich Hickey"}))) (diet {:name "Rich Hickey"}))

View file

@ -1,28 +1,25 @@
(ns koans.11-lazy-sequences
(:require [koan-engine.core :refer :all]))
(meditations ; "There are many ways to generate a sequence"
"There are many ways to generate a sequence"
(= __ (range 1 5)) (= __ (range 1 5))
"The range starts at the beginning by default" ; "The range starts at the beginning by default"
(= __ (range 5)) (= __ (range 5))
"Only take what you need when the sequence is large" ; "Only take what you need when the sequence is large"
(= [0 1 2 3 4 5 6 7 8 9] (= [0 1 2 3 4 5 6 7 8 9]
(take __ (range 100))) (take __ (range 100)))
"Or limit results by dropping what you don't need" ; "Or limit results by dropping what you don't need"
(= [95 96 97 98 99] (= [95 96 97 98 99]
(drop __ (range 100))) (drop __ (range 100)))
"Iteration provides an infinite lazy sequence" ; "Iteration provides an infinite lazy sequence"
(= __ (take 20 (iterate inc 0))) (= __ (take 20 (iterate inc 0)))
"Repetition is key" ; "Repetition is key"
(= [:a :a :a :a :a :a :a :a :a :a] (= [:a :a :a :a :a :a :a :a :a :a]
(repeat 10 __)) (repeat 10 __))
"Iteration can be used for repetition" ; "Iteration can be used for repetition"
(= (repeat 100 :hello) (= (repeat 100 :hello)
(take 100 (iterate ___ :hello)))) (take 100 (iterate ___ :hello)))

View file

@ -1,36 +1,33 @@
(ns koans.12-sequence-comprehensions
(:require [koan-engine.core :refer :all]))
(meditations ; "Sequence comprehensions can bind each element in turn to a symbol"
"Sequence comprehensions can bind each element in turn to a symbol"
(= __ (= __
(for [x (range 6)] (for [x (range 6)]
x)) x))
"They can easily emulate mapping" ; "They can easily emulate mapping"
(= '(0 1 4 9 16 25) (= '(0 1 4 9 16 25)
(map (fn [x] (* x x)) (map (fn [x] (* x x))
(range 6)) (range 6))
(for [x (range 6)] (for [x (range 6)]
__)) __))
"And also filtering" ; "And also filtering"
(= '(1 3 5 7 9) (= '(1 3 5 7 9)
(filter odd? (range 10)) (filter odd? (range 10))
(for [x __ :when (odd? x)] (for [x __ :when (odd? x)]
x)) x))
"Combinations of these transformations is trivial" ; "Combinations of these transformations is trivial"
(= '(1 9 25 49 81) (= '(1 9 25 49 81)
(map (fn [x] (* x x)) (map (fn [x] (* x x))
(filter odd? (range 10))) (filter odd? (range 10)))
(for [x (range 10) :when __] (for [x (range 10) :when __]
__)) __))
"More complex transformations simply take multiple binding forms" ; "More complex transformations simply take multiple binding forms"
(= [[:top :left] [:top :middle] [:top :right] (= [[:top :left] [:top :middle] [:top :right]
[:middle :left] [:middle :middle] [:middle :right] [:middle :left] [:middle :middle] [:middle :right]
[:bottom :left] [:bottom :middle] [:bottom :right]] [:bottom :left] [:bottom :middle] [:bottom :right]]
(for [row [:top :middle :bottom] (for [row [:top :middle :bottom]
column [:left :middle :right]] column [:left :middle :right]]
__))) __))

View file

@ -1,35 +1,30 @@
(ns koans.13-creating-functions
(:require [koan-engine.core :refer :all]))
(defn square [x] (* x x)) ; "One may know what they seek by knowing what they do not seek"
(meditations
"One may know what they seek by knowing what they do not seek"
(= [__ __ __] (let [not-a-symbol? (complement symbol?)] (= [__ __ __] (let [not-a-symbol? (complement symbol?)]
(map not-a-symbol? [:a 'b "c"]))) (map not-a-symbol? [:a 'b "c"])))
"Praise and 'complement' may help you separate the wheat from the chaff" ; "Praise and 'complement' may help you separate the wheat from the chaff"
(= [:wheat "wheat" 'wheat] (= [:wheat "wheat" 'wheat]
(let [not-nil? ___] (let [not-nil? ___]
(filter not-nil? [nil :wheat nil "wheat" nil 'wheat nil]))) (filter not-nil? [nil :wheat nil "wheat" nil 'wheat nil])))
"Partial functions allow procrastination" ; "Partial functions allow procrastination"
(= 20 (let [multiply-by-5 (partial * 5)] (= 20 (let [multiply-by-5 (partial * 5)]
(___ __))) (___ __)))
"Don't forget: first things first" ; "Don't forget: first things first"
(= [__ __ __ __] (= [__ __ __ __]
(let [ab-adder (partial concat [:a :b])] (let [ab-adder (partial concat [:a :b])]
(ab-adder [__ __]))) (ab-adder [__ __])))
"Functions can join forces as one 'composed' function" ; "Functions can join forces as one 'composed' function"
(= 25 (let [inc-and-square (comp square inc)] (= 25 (let [inc-and-square (comp square inc)]
(inc-and-square __))) (inc-and-square __)))
"Have a go on a double dec-er" ; "Have a go on a double dec-er"
(= __ (let [double-dec (comp dec dec)] (= __ (let [double-dec (comp dec dec)]
(double-dec 10))) (double-dec 10)))
"Be careful about the order in which you mix your functions" ; "Be careful about the order in which you mix your functions"
(= 99 (let [square-and-dec ___] (= 99 (let [square-and-dec ___]
(square-and-dec 10)))) (square-and-dec 10)))

View file

@ -1,54 +1,50 @@
(ns koans.14-recursion (defn is-even? [n]
(:require [koan-engine.core :refer :all]))
(defn is-even? [n]
(if (= n 0)
__
(___ (is-even? (dec n)))))
(defn is-even-bigint? [n]
(loop [n n
acc true]
(if (= n 0) (if (= n 0)
__ __
(recur (dec n) (not acc))))) (___ (is-even? (dec n)))))
(defn recursive-reverse [coll] (defn is-even-bigint? [n]
__) (loop [n n
acc true]
(if (= n 0)
__
(recur (dec n) (not acc)))))
(defn factorial [n] (defn recursive-reverse [coll]
__) __)
(meditations (defn factorial [n]
"Recursion ends with a base case" __)
; "Recursion ends with a base case"
(= true (is-even? 0)) (= true (is-even? 0))
"And starts by moving toward that base case" ; "And starts by moving toward that base case"
(= false (is-even? 1)) (= false (is-even? 1))
"Having too many stack frames requires explicit tail calls with recur" ; "Having too many stack frames requires explicit tail calls with recur"
(= false (is-even-bigint? 100003N)) (= false (is-even-bigint? 100003N))
"Reversing directions is easy when you have not gone far" ; "Reversing directions is easy when you have not gone far"
(= '(1) (recursive-reverse [1])) (= '(1) (recursive-reverse [1]))
"Yet it becomes more difficult the more steps you take" ; "Yet it becomes more difficult the more steps you take"
(= '(5 4 3 2 1) (recursive-reverse [1 2 3 4 5])) (= '(5 4 3 2 1) (recursive-reverse [1 2 3 4 5]))
"Simple things may appear simple." ; "Simple things may appear simple."
(= 1 (factorial 1)) (= 1 (factorial 1))
"They may require other simple steps." ; "They may require other simple steps."
(= 2 (factorial 2)) (= 2 (factorial 2))
"Sometimes a slightly bigger step is necessary" ; "Sometimes a slightly bigger step is necessary"
(= 6 (factorial 3)) (= 6 (factorial 3))
"And eventually you must think harder" ; "And eventually you must think harder"
(= 24 (factorial 4)) (= 24 (factorial 4))
"You can even deal with very large numbers" ; "You can even deal with very large numbers"
(< 1000000000000000000000000N (factorial 1000N)) (< 1000000000000000000000000N (factorial 1000N))
"But what happens when the machine limits you?" ; "But what happens when the machine limits you?"
(< 1000000000000000000000000N (factorial 100003N))) (< 1000000000000000000000000N (factorial 100003N))

View file

@ -1,44 +1,36 @@
(ns koans.15-destructuring
(:require [koan-engine.core :refer :all]))
(def test-address ; "Destructuring is an arbiter: it breaks up arguments"
{:street-address "123 Test Lane"
:city "Testerville"
:state "TX"})
(meditations
"Destructuring is an arbiter: it breaks up arguments"
(= __ ((fn [[a b]] (str b a)) (= __ ((fn [[a b]] (str b a))
[:foo :bar])) [:foo :bar]))
"Whether in function definitions" ; "Whether in function definitions"
(= (str "An Oxford comma list of apples, " (= (str "An Oxford comma list of apples, "
"oranges, " "oranges, "
"and pears.") "and pears.")
((fn [[a b c]] __) ((fn [[a b c]] __)
["apples" "oranges" "pears"])) ["apples" "oranges" "pears"]))
"Or in let expressions" ; "Or in let expressions"
(= "Rich Hickey aka The Clojurer aka Go Time aka Lambda Guru" (= "Rich Hickey aka The Clojurer aka Go Time aka Lambda Guru"
(let [[first-name last-name & aliases] (let [[first-name last-name & aliases]
(list "Rich" "Hickey" "The Clojurer" "Go Time" "Lambda Guru")] (list "Rich" "Hickey" "The Clojurer" "Go Time" "Lambda Guru")]
__)) __))
"You can regain the full argument if you like arguing" ; "You can regain the full argument if you like arguing"
(= {:original-parts ["Stephen" "Hawking"] :named-parts {:first "Stephen" :last "Hawking"}} (= {:original-parts ["Stephen" "Hawking"] :named-parts {:first "Stephen" :last "Hawking"}}
(let [[first-name last-name :as full-name] ["Stephen" "Hawking"]] (let [[first-name last-name :as full-name] ["Stephen" "Hawking"]]
__)) __))
"Break up maps by key" ; "Break up maps by key"
(= "123 Test Lane, Testerville, TX" (= "123 Test Lane, Testerville, TX"
(let [{street-address :street-address, city :city, state :state} test-address] (let [{street-address :street-address, city :city, state :state} test-address]
__)) __))
"Or more succinctly" ; "Or more succinctly"
(= "123 Test Lane, Testerville, TX" (= "123 Test Lane, Testerville, TX"
(let [{:keys [street-address __ __]} test-address] (let [{:keys [street-address __ __]} test-address]
__)) __))
"All together now!" ; "All together now!"
(= "Test Testerson, 123 Test Lane, Testerville, TX" (= "Test Testerson, 123 Test Lane, Testerville, TX"
(___ ["Test" "Testerson"] test-address))) (___ ["Test" "Testerson"] test-address))

View file

@ -1,22 +1,16 @@
(ns koans.16-refs
(:require [koan-engine.core :refer :all]))
(def the-world (ref "hello")) ; "In the beginning, there was a word"
(def bizarro-world (ref {}))
(meditations
"In the beginning, there was a word"
(= __ (deref the-world)) (= __ (deref the-world))
"You can get the word more succinctly, but it's the same" ; "You can get the word more succinctly, but it's the same"
(= __ @the-world) (= __ @the-world)
"You can be the change you wish to see in the world." ; "You can be the change you wish to see in the world."
(= __ (do (= __ (do
(dosync (ref-set the-world "better")) (dosync (ref-set the-world "better"))
@the-world)) @the-world))
"Alter where you need not replace" ; "Alter where you need not replace"
(= __ (let [exclamator (fn [x] (str x "!"))] (= __ (let [exclamator (fn [x] (str x "!"))]
(dosync (dosync
(alter the-world exclamator) (alter the-world exclamator)
@ -24,19 +18,19 @@
(alter the-world exclamator)) (alter the-world exclamator))
@the-world)) @the-world))
"Don't forget to do your work in a transaction!" ; "Don't forget to do your work in a transaction!"
(= 0 (do __ (= 0 (do __
@the-world)) @the-world))
"Functions passed to alter may depend on the data in the ref" ; "Functions passed to alter may depend on the data in the ref"
(= 20 (do (= 20 (do
(dosync (alter the-world ___)))) (dosync (alter the-world ___))))
"Two worlds are better than one" ; "Two worlds are better than one"
(= ["Real Jerry" "Bizarro Jerry"] (= ["Real Jerry" "Bizarro Jerry"]
(do (do
(dosync (dosync
(ref-set the-world {}) (ref-set the-world {})
(alter the-world assoc :jerry "Real Jerry") (alter the-world assoc :jerry "Real Jerry")
(alter bizarro-world assoc :jerry "Bizarro Jerry") (alter bizarro-world assoc :jerry "Bizarro Jerry")
__)))) __)))

View file

@ -1,33 +1,28 @@
(ns koans.17-atoms
(:require [koan-engine.core :refer :all]))
(def atomic-clock (atom 0)) ; "Atoms are like refs"
(meditations
"Atoms are like refs"
(= __ @atomic-clock) (= __ @atomic-clock)
"You can change at the swap meet" ; "You can change at the swap meet"
(= __ (do (= __ (do
(swap! atomic-clock inc) (swap! atomic-clock inc)
@atomic-clock)) @atomic-clock))
"Keep taxes out of this: swapping requires no transaction" ; "Keep taxes out of this: swapping requires no transaction"
(= 5 (do (= 5 (do
__ __
@atomic-clock)) @atomic-clock))
"Any number of arguments might happen during a swap" ; "Any number of arguments might happen during a swap"
(= __ (do (= __ (do
(swap! atomic-clock + 1 2 3 4 5) (swap! atomic-clock + 1 2 3 4 5)
@atomic-clock)) @atomic-clock))
"Atomic atoms are atomic" ; "Atomic atoms are atomic"
(= __ (do (= __ (do
(compare-and-set! atomic-clock 100 :fin) (compare-and-set! atomic-clock 100 :fin)
@atomic-clock)) @atomic-clock))
"When your expectations are aligned with reality, things proceed that way" ; "When your expectations are aligned with reality, things proceed that way"
(= :fin (do (= :fin (do
(compare-and-set! __ __ __) (compare-and-set! __ __ __)
@atomic-clock))) @atomic-clock))

View file

@ -1,5 +1,3 @@
(ns koans.18-macros
(:require [koan-engine.core :refer :all]))
(defmacro hello [x] (defmacro hello [x]
(str "Hello, " x)) (str "Hello, " x))
@ -25,21 +23,20 @@
(r-infix ~first-arg) (r-infix ~first-arg)
(r-infix ~others))))) (r-infix ~others)))))
(meditations ; "Macros are like functions created at compile time"
"Macros are like functions created at compile time"
(= __ (hello "Macros!")) (= __ (hello "Macros!"))
"I can haz infix?" ; "I can haz infix?"
(= __ (infix (9 + 1))) (= __ (infix (9 + 1)))
"Remember, these are nothing but code transformations" ; "Remember, these are nothing but code transformations"
(= __ (macroexpand '(infix (9 + 1)))) (= __ (macroexpand '(infix (9 + 1))))
"You can do better than that - hand crafting FTW!" ; "You can do better than that - hand crafting FTW!"
(= __ (macroexpand '(infix-better (10 * 2)))) (= __ (macroexpand '(infix-better (10 * 2))))
"Things don't always work as you would like them to... " ; "Things don't always work as you would like them to... "
(= __ (macroexpand '(infix-better ( 10 + (2 * 3))))) (= __ (macroexpand '(infix-better ( 10 + (2 * 3)))))
"Really, you don't understand recursion until you understand recursion" ; "Really, you don't understand recursion until you understand recursion"
(= 36 (r-infix (10 + (2 * 3) + (4 * 5))))) (= 36 (r-infix (10 + (2 * 3) + (4 * 5))))

View file

@ -1,47 +1,25 @@
(ns koans.19-datatypes
(:require [koan-engine.core :refer :all]))
(defrecord Nobel [prize]) ; "Holding records is meaningful only when the record is worthy of you"
(deftype Pulitzer [prize])
(defprotocol Award
(present [this recipient]))
(defrecord Oscar [category]
Award
(present [this recipient]
(print (str "Congratulations on your "
(:category this) " Oscar, "
recipient
"!"))))
(deftype Razzie [category]
Award
(present [this recipient]
__))
(meditations
"Holding records is meaningful only when the record is worthy of you"
(= __ (.prize (Nobel. "peace"))) (= __ (.prize (Nobel. "peace")))
"Types are quite similar" ; "Types are quite similar"
(= __ (.prize (Pulitzer. "literature"))) (= __ (.prize (Pulitzer. "literature")))
"Records may be treated like maps" ; "Records may be treated like maps"
(= __ (:prize (Nobel. "physics"))) (= __ (:prize (Nobel. "physics")))
"While types may not" ; "While types may not"
(= __ (:prize (Pulitzer. "poetry"))) (= __ (:prize (Pulitzer. "poetry")))
"Further study reveals why" ; "Further study reveals why"
(= __ (= __
(map map? [(Nobel. "chemistry") (map map? [(Nobel. "chemistry")
(Pulitzer. "music")])) (Pulitzer. "music")]))
"Either sort of datatype can define methods in a protocol" ; "Either sort of datatype can define methods in a protocol"
(= __ (= __
(with-out-str (present (Oscar. "Best Picture") "Evil Alien Conquerors"))) (with-out-str (present (Oscar. "Best Picture") "Evil Alien Conquerors")))
"Surely we can implement our own by now" ; "Surely we can implement our own by now"
(= "You're really the Worst Picture, Final Destination 5... sorry." (= "You're really the Worst Picture, Final Destination 5... sorry."
(with-out-str (present (Razzie. "Worst Picture") "Final Destination 5")))) (with-out-str (present (Razzie. "Worst Picture") "Final Destination 5")))

View file

@ -1,19 +1,16 @@
(ns koans.20-java-interop
(:require [koan-engine.core :refer :all]))
(meditations ; "You may have done more with Java than you know"
"You may have done more with Java than you know"
(= __ (class "warfare")) ; hint: try typing (javadoc "warfare") in the REPL (= __ (class "warfare")) ; hint: try typing (javadoc "warfare") in the REPL
"The dot signifies easy and direct Java interoperation" ; "The dot signifies easy and direct Java interoperation"
(= __ (.toUpperCase "select * from")) (= __ (.toUpperCase "select * from"))
"But instance method calls are very different from normal functions" ; "But instance method calls are very different from normal functions"
(= ["SELECT" "FROM" "WHERE"] (map ___ ["select" "from" "where"])) (= ["SELECT" "FROM" "WHERE"] (map ___ ["select" "from" "where"]))
"Constructing might be harder than breaking" ; "Constructing might be harder than breaking"
(= 10 (let [latch (java.util.concurrent.CountDownLatch. __)] (= 10 (let [latch (java.util.concurrent.CountDownLatch. __)]
(.getCount latch))) (.getCount latch)))
"Static methods are slashing prices!" ; "Static methods are slashing prices!"
(== __ (Math/pow 2 10))) (== __ (Math/pow 2 10))

View file

@ -1,21 +1,18 @@
(ns koans.21-partition
(:require [koan-engine.core :refer :all]))
(meditations ; "To split a collection you can use the partition function"
"To split a collection you can use the partition function"
(= '((0 1) (2 3)) (__ 2 (range 4))) (= '((0 1) (2 3)) (__ 2 (range 4)))
"But watch out if there are not enough elements to form n sequences" ; "But watch out if there are not enough elements to form n sequences"
(= '(__) (partition 3 [:a :b :c :d :e])) (= '(__) (partition 3 [:a :b :c :d :e]))
"You can use partition-all to include any leftovers too" ; "You can use partition-all to include any leftovers too"
(= __ (partition-all 3 (range 5))) (= __ (partition-all 3 (range 5)))
"If you need to, you can start each sequence with an offset" ; "If you need to, you can start each sequence with an offset"
(= '((0 1 2) (5 6 7) (10 11 12)) (partition 3 __ (range 13))) (= '((0 1 2) (5 6 7) (10 11 12)) (partition 3 __ (range 13)))
"Consider padding the last sequence with some default values..." ; "Consider padding the last sequence with some default values..."
(= '((0 1 2) (3 4 5) (6 :hello)) (partition 3 3 [__] (range 7))) (= '((0 1 2) (3 4 5) (6 :hello)) (partition 3 3 [__] (range 7)))
"... but notice that they will only pad up to the given sequence length" ; "... but notice that they will only pad up to the given sequence length"
(= '((0 1 2) (3 4 5) __) (partition 3 3 [:these :are "my" "words"] (range 7)))) (= '((0 1 2) (3 4 5) __) (partition 3 3 [:these :are "my" "words"] (range 7)))

View file

@ -1,26 +1,19 @@
(ns koans.22-group-by
(:require [koan-engine.core :refer :all]))
(defn get-odds-and-evens [coll] ; "To categorize a collection by some function, use group-by."
(let [{odds true evens false} (group-by __ coll)]
[odds evens]))
(meditations
"To categorize a collection by some function, use group-by."
(= __ (group-by count ["hello" "world" "foo" "bar"])) (= __ (group-by count ["hello" "world" "foo" "bar"]))
"You can simulate filter + remove in one pass" ; "You can simulate filter + remove in one pass"
(= (get-odds-and-evens [1 2 3 4 5]) (= (get-odds-and-evens [1 2 3 4 5])
((juxt filter remove) odd? [1 2 3 4 5]) ((juxt filter remove) odd? [1 2 3 4 5])
[[1 3 5] [2 4]]) [[1 3 5] [2 4]])
"You can also group by a primary key" ; "You can also group by a primary key"
(= __ (= __
(group-by :id [{:id 1 :name "Bob"} (group-by :id [{:id 1 :name "Bob"}
{:id 2 :name "Jennifer"} {:id 2 :name "Jennifer"}
{:id 1 :last-name "Smith"} ])) {:id 1 :last-name "Smith"} ]))
"But be careful when you group by a non-required key" ; "But be careful when you group by a non-required key"
(= {"Bob" [{:name "Bob" :id 1}] (= {"Bob" [{:name "Bob" :id 1}]
"Jennifer" [{:name "Jennifer" :id 2}] "Jennifer" [{:name "Jennifer" :id 2}]
__ [{:last-name "Smith" :id 1}]} __ [{:last-name "Smith" :id 1}]}
@ -28,9 +21,9 @@
{:id 2 :name "Jennifer"} {:id 2 :name "Jennifer"}
{:id 1 :last-name "Smith"}])) {:id 1 :last-name "Smith"}]))
"The true power of group-by comes with custom functions" ; "The true power of group-by comes with custom functions"
(= __ (= __
(group-by #(if (:bad %) :naughty-list :nice-list) (group-by #(if (:bad %) :naughty-list :nice-list)
[{:name "Jimmy" :bad true} [{:name "Jimmy" :bad true}
{:name "Jane" :bad false} {:name "Jane" :bad false}
{:name "Joe" :bad true}]))) {:name "Joe" :bad true}]))

View file

@ -1,18 +1,11 @@
(ns koans.23-meta
(:require [koan-engine.core :refer :all]))
(def giants ; "Some objects can be tagged using the with-meta function"
(with-meta 'Giants
{:league "National League"}))
(meditations
"Some objects can be tagged using the with-meta function"
(= __ (meta giants)) (= __ (meta giants))
"Or more succinctly with a reader macro" ; "Or more succinctly with a reader macro"
(= __ (meta '^{:division "West"} Giants)) (= __ (meta '^{:division "West"} Giants))
"While others can't" ; "While others can't"
(= __ (try (= __ (try
(with-meta (with-meta
2 2
@ -20,18 +13,18 @@
(catch ClassCastException e (catch ClassCastException e
"This doesn't implement the IObj interface"))) "This doesn't implement the IObj interface")))
"Notice when metadata carries over" ; "Notice when metadata carries over"
(= __ (meta (merge '^{:foo :bar} {:a 1 :b 2} (= __ (meta (merge '^{:foo :bar} {:a 1 :b 2}
{:b 3 :c 4}))) {:b 3 :c 4})))
"And when it doesn't" ; "And when it doesn't"
(= __ (meta (merge {:a 1 :b 2} (= __ (meta (merge {:a 1 :b 2}
'^{:foo :bar} {:b 3 :c 4}))) '^{:foo :bar} {:b 3 :c 4})))
"Metadata can be used as a type hint to avoid reflection during runtime" ; "Metadata can be used as a type hint to avoid reflection during runtime"
(= __ (#(.charAt ^String % 0) "Cast me")) (= __ (#(.charAt ^String % 0) "Cast me"))
"You can directly update an object's metadata" ; "You can directly update an object's metadata"
(= 8 (let [giants (= 8 (let [giants
(with-meta (with-meta
'Giants 'Giants
@ -39,13 +32,13 @@
(swap! (:world-series-titles (meta giants)) __) (swap! (:world-series-titles (meta giants)) __)
@(:world-series-titles (meta giants)))) @(:world-series-titles (meta giants))))
"You can also create a new object from another object with metadata" ; "You can also create a new object from another object with metadata"
(= {:league "National League" :park "AT&T Park"} (= {:league "National League" :park "AT&T Park"}
(meta (vary-meta giants (meta (vary-meta giants
assoc __ __))) assoc __ __)))
"But it won't affect behavior like equality" ; "But it won't affect behavior like equality"
(= __ (vary-meta giants dissoc :league)) (= __ (vary-meta giants dissoc :league))
"Or the object's printed representation" ; "Or the object's printed representation"
(= __ (pr-str (vary-meta giants dissoc :league)))) (= __ (pr-str (vary-meta giants dissoc :league)))