This is the first 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 meetup) | |
(defn leap? [year] | |
(let | |
[divisible-by? (comp zero? (partial rem year))] | |
(or | |
(divisible-by? 400) | |
(and | |
(divisible-by? 4) | |
(not (divisible-by? 100)))))) | |
(defn get-months-accum-days-moduli [year month] | |
(let | |
[months-accum-days-moduli-year | |
[0, 3, 3, 6, 1, 4, 6, 2, 5, 0, 3, 5] | |
months-accum-days-moduli-leap-year | |
[0, 3, 4, 0, 2, 5, 0, 3, 6, 1, 4, 6]] | |
(if (leap? year) | |
(nth months-accum-days-moduli-leap-year (- month 1)) | |
(nth months-accum-days-moduli-year (- month 1))))) | |
(defrecord Date [year month day]) | |
(defn compute-week-day [{year :year month :month day :day}] | |
(let | |
[div1 (quot (- year 1) 4) | |
div2 (quot (- year 1) 100) | |
div3 (quot (+ div2 1) 4) | |
div4 (int (- div1 (* 3 div3))) | |
week-day (rem | |
(+ (rem (- year 1) 7) | |
(rem div4 7) | |
(get-months-accum-days-moduli year month) | |
(rem day 7)) | |
7)] | |
(if (zero? week-day) 7 week-day))) | |
(defn last-day-in-month [month year] | |
(let | |
[last-days-in-months | |
[31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31] | |
last-day (nth last-days-in-months (- month 1))] | |
(if (and (leap? year) (= month 2)) | |
(+ 1 last-day) | |
last-day))) | |
(defn dates-in-month [month year] | |
(map (fn [day] (Date. year month day)) | |
(range 1 (+ (last-day-in-month month year) 1)))) | |
(defn is-week-day? [week-day date] | |
(let | |
[week-days {:MON 1 :TUE 2 :WED 3 :THU 4 :FRI 5 :SAT 6 :SUN 7}] | |
(= (week-days week-day) (compute-week-day date)))) | |
(defn is-teenth? [{day :day}] | |
(let | |
[teenth-days (set (range 13 20))] | |
(teenth-days day))) | |
(defn days-of-month-that-is [week-day] | |
(comp | |
(partial | |
filter | |
(partial is-week-day? week-day)) | |
dates-in-month)) | |
(defn day-teenth [week-day] | |
(comp vec | |
vals | |
first | |
(partial filter #(is-teenth? %)) | |
(days-of-month-that-is week-day))) | |
(def monteenth (day-teenth :MON)) | |
(def tuesteenth (day-teenth :TUE)) | |
(def wednesteenth (day-teenth :WED)) | |
(def thursteenth (day-teenth :THU)) | |
(def friteenth (day-teenth :FRI)) | |
(def saturteenth (day-teenth :SAT)) | |
(def sunteenth (day-teenth :SUN)) | |
(defn get-week-day [pos week-day] | |
(comp vec | |
vals | |
pos | |
(days-of-month-that-is week-day))) | |
(def first-monday (get-week-day first :MON)) | |
(def first-tuesday (get-week-day first :TUE)) | |
(def first-wednesday (get-week-day first :WED)) | |
(def first-thursday (get-week-day first :THU)) | |
(def first-friday (get-week-day first :FRI)) | |
(def first-saturday (get-week-day first :SAT)) | |
(def first-sunday (get-week-day first :SUN)) | |
(def second-monday (get-week-day second :MON)) | |
(def second-tuesday (get-week-day second :TUE)) | |
(def second-wednesday (get-week-day second :WED)) | |
(def second-thursday (get-week-day second :THU)) | |
(def second-friday (get-week-day second :FRI)) | |
(def second-saturday (get-week-day second :SAT)) | |
(def second-sunday (get-week-day second :SUN)) | |
(defn third [ls] (nth ls 2)) | |
(def third-monday (get-week-day third :MON)) | |
(def third-tuesday (get-week-day third :TUE)) | |
(def third-wednesday (get-week-day third :WED)) | |
(def third-thursday (get-week-day third :THU)) | |
(def third-friday (get-week-day third :FRI)) | |
(def third-saturday (get-week-day third :SAT)) | |
(def third-sunday (get-week-day third :SUN)) | |
(defn fourth [ls] (nth ls 3)) | |
(def fourth-monday (get-week-day fourth :MON)) | |
(def fourth-tuesday (get-week-day fourth :TUE)) | |
(def fourth-wednesday (get-week-day fourth :WED)) | |
(def fourth-thursday (get-week-day fourth :THU)) | |
(def fourth-friday (get-week-day fourth :FRI)) | |
(def fourth-saturday (get-week-day fourth :SAT)) | |
(def fourth-sunday (get-week-day fourth :SUN)) | |
(def last-monday (get-week-day last :MON)) | |
(def last-tuesday (get-week-day last :TUE)) | |
(def last-wednesday (get-week-day last :WED)) | |
(def last-thursday (get-week-day last :THU)) | |
(def last-friday (get-week-day last :FRI)) | |
(def last-saturday (get-week-day last :SAT)) | |
(def last-sunday (get-week-day last :SUN)) |
To make it more interesting I decided to implement the date logic myself instead of using a Date library.
What I really didn't like about this version was that I had to generate one by one a lot of functions.
I commented in my solution that I wasn't happy with the result and that I would love to find out a way to dynamically generate all the functions that the tests were using.
Yesterday I received a nitpick from moog just saying:
"intern is your friend"So I started to google a way to use intern to solve my problem. After a while I found this post: Metaprogramming with Clojure explaining how to use intern to dynamically generate bindings in a given namespace.
I had to do several trials in the REPL and remember to force the evaluation of a sequence with doall before getting it to work using map and a list comprehension.
This is the new version in which I managed to remove the clutter by dynamically generating all the functions that the tests use (look at the two calls to doall nearly at the end of the code). It is around 30 lines shorter:
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 meetup) | |
(defn leap? [year] | |
(let | |
[divisible-by? (comp zero? (partial rem year))] | |
(or | |
(divisible-by? 400) | |
(and | |
(divisible-by? 4) | |
(not (divisible-by? 100)))))) | |
(defn get-months-accum-days-moduli [year month] | |
(let | |
[months-accum-days-moduli-year | |
[0, 3, 3, 6, 1, 4, 6, 2, 5, 0, 3, 5] | |
months-accum-days-moduli-leap-year | |
[0, 3, 4, 0, 2, 5, 0, 3, 6, 1, 4, 6]] | |
(if (leap? year) | |
(nth months-accum-days-moduli-leap-year (- month 1)) | |
(nth months-accum-days-moduli-year (- month 1))))) | |
(defrecord Date [year month day]) | |
(defn compute-week-day [{year :year month :month day :day}] | |
(let | |
[div1 (quot (- year 1) 4) | |
div2 (quot (- year 1) 100) | |
div3 (quot (+ div2 1) 4) | |
div4 (int (- div1 (* 3 div3))) | |
week-day (rem | |
(+ (rem (- year 1) 7) | |
(rem div4 7) | |
(get-months-accum-days-moduli year month) | |
(rem day 7)) | |
7)] | |
(if (zero? week-day) 7 week-day))) | |
(defn last-day-in-month [month year] | |
(let | |
[last-days-in-months | |
[31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31] | |
last-day (nth last-days-in-months (- month 1))] | |
(if (and (leap? year) (= month 2)) | |
(+ 1 last-day) | |
last-day))) | |
(defn dates-in-month [month year] | |
(map (fn [day] (Date. year month day)) | |
(range 1 (+ (last-day-in-month month year) 1)))) | |
(def week-days {'mon 1 'tues 2 'wednes 3 'thurs 4 'fri 5 'satur 6 'sun 7}) | |
(defn is-week-day? [week-day date] | |
(= (week-days week-day) (compute-week-day date))) | |
(defn is-teenth? [{day :day}] | |
(let | |
[teenth-days (set (range 13 20))] | |
(teenth-days day))) | |
(defn days-of-month-that-is [week-day] | |
(comp | |
(partial | |
filter | |
(partial is-week-day? week-day)) | |
dates-in-month)) | |
(defn day-teenth [week-day] | |
(comp vec | |
vals | |
first | |
(partial filter #(is-teenth? %)) | |
(days-of-month-that-is week-day))) | |
(defn third [ls] (nth ls 2)) | |
(defn fourth [ls] (nth ls 3)) | |
(defn get-week-day [pos week-day] | |
(comp vec | |
vals | |
pos | |
(days-of-month-that-is week-day))) | |
(doall | |
(map | |
#(intern 'meetup (symbol (str (name %) "teenth")) (day-teenth %)) | |
(keys week-days))) | |
(def func-names ["first" "second" "third" "fourth" "last"]) | |
(doall | |
(map | |
#(intern | |
'meetup | |
(symbol (str (first %) "-" (name (second %)) "day")) | |
(get-week-day (resolve (symbol(first %))) (second %))) | |
(for [func-name func-names | |
week-day (keys week-days)] | |
[func-name week-day]))) |
I'm glad because I've learned about a lot of new Clojure functions, symbol, resolve, name, intern and doall, and also used a list comprehension with for.
I'd like to thank moog for giving me a passage into a new Clojure territory.
You can nitpick my solution here or see all the exercises I've done so far in this repository.
No comments:
Post a Comment