This is my solution for an exercise from the first chapter, Just Enough Clojure in which I had to write a "bizarre version of factorial that uses neither iteration nor 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
(defn bizarre-factorial [n] | |
(apply * (range 1 (+ n 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
user=> (bizarre-factorial 1) | |
1 | |
user=> (bizarre-factorial 2) | |
2 | |
user=> (bizarre-factorial 3) | |
6 | |
user=> (bizarre-factorial 4) | |
24 | |
user=> (bizarre-factorial 5) | |
120 | |
user=> (bizarre-factorial 6) | |
720 | |
user=> (bizarre-factorial 7) | |
5040 | |
user=> (bizarre-factorial 10) | |
3628800 |
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
user=> (bizarre-factorial 40) | |
ArithmeticException integer overflow | |
clojure.lang.Numbers.throwIntOverflow (Numbers.java:1388) |
You just need to append an N after any number in the calculation, since it's "contagious" to the rest of the calculation:
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 bizarre-factorial [n] | |
(apply * (range 1N (+ n 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
user=> (bizarre-factorial 40) | |
815915283247897734345611269596115894272000000000N | |
user=> (bizarre-factorial 100) | |
93326215443944152681699238856266700490715968264381621468592963895 | |
21759999322991560894146397615651828625369792082722375825118521091 | |
6864000000000000000000000000N |
That's ok but how does bizarre-factorial works?
Let's first examine range function:
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
user=> (doc range) | |
------------------------- | |
clojure.core/range | |
([] [end] [start end] [start end step]) | |
Returns a lazy seq of nums from start (inclusive) to end | |
(exclusive), by step, where start defaults to 0, step to 1, | |
and end to infinity. | |
nil |
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
user=> (range 1N 11) | |
(1N 2N 3N 4N 5N 6N 7N 8N 9N 10N) |
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 bizarre-factorial [n] | |
(let [positive-ints-until (fn [n] (range 1N (+ n 1)))] | |
(apply * (positive-ints-until 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
user=> (doc apply) | |
------------------------- | |
clojure.core/apply | |
([f args] [f x args] [f x y args] [f x y z args] [f a b c d & args]) | |
Applies fn f to the argument list formed by prepending | |
intervening arguments to args. | |
nil |
That's how bizarre-factorial works.
Finally we can use partial to give a better name to the partial application of apply and *:
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 bizarre-factorial [n] | |
(let [positive-ints-until (fn [n] (range 1N (+ n 1))) | |
multiply (partial apply *)] | |
(multiply (positive-ints-until n)))) |
No comments:
Post a Comment