Contents
The brainwave here is to (mis)use the feature set of Clojure and its standard library to cook up as many ways to encode FizzBuzz as one can muster (or steal). If all goes well, this post will receive many updates. If it goes really well, all sorts of bad ideas and clever foot-guns will be discovered and used.
The creative constraint is this: any FizzBuzz, however terrible or hilarious, must also be useful. It should have reason to exist and should reveal some real-world Clojure thinking.
That means no FizzBuzzEnterpriseEdition but also no Dylan Beattie's brilliant FizzBuzz in RockStar. So sorry!
Not to be indelicate, but I will state the problem before proceeding.
Fizz buzz is a group word game for children to teach them about division. Players take turns to count incrementally, replacing any number divisible by three with the word "fizz", and any number divisible by five with the word "buzz".
Needless to say, they mean natural numbers starting at 1 when they say "numbers".
Phew that was rough on the ego. Let us compose ourselves for a minute.
…
OK, onward.
Usage guide
Reading the code
Find the
check-all-fizz-buzzers
function below, for usage examples.We use functions only from Clojure's standard library (clojure.core).
Lookup unfamiliar functions at https://clojuredocs.org/quickref, helpfully illustrated with community-contributed examples.
Running the code
If you don't already have Clojure installed, you can run the code online at https://tryclojure.org/ (fully client-side, but minimalist), or at https://repl.it/languages/clojure (feature-rich web-based IDE, but requires signup).
Preferably, copy over code as you go along. Try it out bit by bit.
If you are in a hurry, find the "Interlude" section and copy over all the code from there.
Finally, run
(check-all-fizz-buzzers)
to see if it all works as expected.
Complaining about the code
- If I've made bugs in the Code or the English (possible) or made egregious remarks (very possible), please write to weblog at this website. I'll fix the bugs.
Right, then. Shall we begin?
Le FizzBuzz Classique
In the beginning, one might trawl the Clojure standard library for familiar, safe-looking words and accidentally discover for
1. Python or Javascript gentlenerds may say "Ooh, List Comprehension. Nice!", and bang out Le FizzBuzz Classique. Java or C# gentlenerds may struggle a lot more, because Clojure has no Class
. We are sorry for this disappointment. Please follow the Python for now.
defn fizz-buzz-classic
(
[num-xs]for [n num-xs]
(cond
(zero? (rem n 15)) (println "FizzBuzz")
(zero? (rem n 3)) (println "Fizz")
(zero? (rem n 5)) (println "Buzz")
(:else (println n))))
Then you will evaluate it in the "REPL", et voilà! Something très familiar!
list 1 3 5 15 16))
user=> (fizz-buzz-classic (1
Fizz
Buzz
FizzBuzz16
nil nil nil nil nil) (
(Except for that pesky last line full of nils. But like, whatever. It worked.).
Le FizzBuzz Classique est mort à Clojure. Désolé :(
Sadly, the spurious nils are the least of your woes. You just fell prey to something called "Laziness", and the code is dead on arrival, but you don't know it yet because evaluating in the "REPL" obscures this fact.
Welcome to Functional Programming (FP) with lazy sequences, which is awesome, but which is also one of the double edges of Clojure that will cut you if you come here with set ideas about How Things Ought To Work.
Saying it in French (however broken) felt gentler, somehow.
At this point, you might accuse me of setting you up with this strawman for
. In response, I might simply wait for your production to blow up. Unknowing mixing of laziness and side effects reliably trips up all programmers new to Clojure 2.
Luckily we can avoid going down that rabbit hole 3 entirely, because there is a more pressing problem that makes Le FizzBuzz Classique look severely defective to this Clojurist's FP-addled brain. Fixing that makes the point of lazy sequences moot, as a nice bonus.
Le FizzBuzz Classique, remedied
Behold this cleaned up version.
defn lazybuzz
(
[num-xs]for [n num-xs]
(cond
(zero? (rem n 15)) "FizzBuzz"
(zero? (rem n 3)) "Fizz"
(zero? (rem n 5)) "Buzz"
(:else n)))
Yes, println
is no more, and…
list 1 3 5 15 16))
user=> (lazybuzz (1 "Fizz" "Buzz" "FizzBuzz" 16) (
Compare the shape of the collection of nils seen classically, with what we see now. They are both sequences with the same number of items, but the new one contains useful values (insert :trollface: :).
list 1 3 5 15 19))
user=> (fizz-buzz-classic (1
Fizz
Buzz
FizzBuzz19
nil nil nil nil nil) ; 5 prints, 5 nils
(
list 1 3 5 15 19))
user=> (lazybuzz (1 "Fizz" "Buzz" "FizzBuzz" 19) ; no prints, 1 value containing 5 values (
Well, that's because all expressions in Clojure return a value. println
creates a side effect of printing and has a return value of nil
. Thus for each item in the input range, the "impure" classic version prints to the REPL, collects the return value of each println (nil), and returns that collection.
The "purified" fizz-buzz simply calculates a result for each branch and the for
returns the accumulated result. And now the results are printed inside parentheses, which is, like, sure whatever. At least it looks like it's doing the right calculations and the pesky nils are gone, so we can move on?
Not so fast.
Le FizzBuzz Classique, dissected
To FizzBuzz creatively in Clojure, we must examine and avoid the defects of the classic version, which are as follows.
- Broken behaviour:
println
alters the state of the world and thus injects non-determinism into an otherwise purely functional calculation. This is anathema 4 to Clojurists (and FP practitioners at large). - Broken API contract: We get back a useless collection nils, instead of the result of a calculation that we can use further. We prefer to always write functions that return useful values.
- Broken time model: Effects want to happen "now" (here, printing to some output device), while lazy computations want to happen "maybe never" (here, a definition that maps the domain of a collection of countless numbers to the domain of FizzBuzz). Effects and laziness can be made to pair well, but only when we define them separately from the get go, and have some third way of joining them together safely when needed. For now, you could do worse than lasering this into your brain: "Never mix (side) effects and laziness.".
- Broken aesthetic: We like our functions to do one job, and do it well. Printing things is a second job, and as Messers Hal and Gerry like to say in SICP 5, "That's George's problem.".
Henceforth, all functions shall be pure calculations, and we will rely on our metaphorical George "the REPL" Ableman to handle all our print jobs.
As an added benefit, writing pure functions makes laziness such a good friend, that we don't even need to acknowledge its presence.
Yet another benefit is that we won't have to burn hundreds of words to apologise for broken code 6.
See? Such passion. I wasn't joking when I said "looks severely defective to me" 7.
Le FizzBuzz Classique, doseq'd
Maybe you still aren't convinced. println
is such a global standard. Easy 8. So you might dig into the standard library more and come up with doseq
to eliminate laziness…
defn doseq-fizz-buzz
(
[num-xs]doseq [n num-xs]
(cond
(zero? (rem n 15)) (println "FizzBuzz")
(zero? (rem n 3)) (println "Fizz")
(zero? (rem n 5)) (println "Buzz")
(:else (println n))))
And declare victory…
list 1 3 5 15 19))
user=> (doseq-fizz-buzz (1
Fizz
Buzz
FizzBuzz19
nil ; maybe we can live with just one nil?
But the code is still fatally broken for the other reasons, and now it is also worse, because this implementation cannot say "here are all the fizzbuzzes". Only a lazy definition can say this and allow you to carry on computing. Besides, doseq
is meant for cases when we want to cause side effects. And the functional Clojurist almost never wants to.
Remember the children's game definition of FizzBuzz? It is beautiful because it does not say "FizzBuzz only for the first K numbers". Now if you go DuckDuck search the standard coding interview version of the question, what do you find? "Write a program that prints the numbers from 1 to 100 such that…".
Boo.
Le FizzBuzz Classique, doall'd
As a famous TV detective would say; "Oh, just one other thing.". Here are ways to break your programs. And if you are feeling suitably adventurous, to also test the stability of your employment.
The following invocation of lazybuzz
in your -main
would not be fine, assuming you wanted to do something useful with it. But would also not precipitate anything terrible.
defn -main
("The entry point to your microservice."
[& args];; Do things ...
println "I'm about to do...")
(
;; No block, no foul.
range))
(lazybuzz (
;; Sure, why not ...
println "I'm done!")) (
Here is a good way to break your software and print FizzBuzzes to the console indefinitely (or at least as long as your computer can make new numbers).
defn -main
("The entry point to your microservice."
[& args];; Do things ...
println "I'm about to do...")
(
;; Spin wheels until the numbers run out.
range))
(doseq-fizz-buzz (
;; Maybe never ...
println "I'm done!")) (
As a funner party trick, if you make a computer (VM) with a bad output device (or redirect program output to /dev/full), then you can crash or hang your program. If you discover it crashes, feel free to daemonise it and make an infinitely restarting JVM process that does nothing but burn CPU cycles. Take that, cryptominers!
To be fair, you can also break programs with lazy evaluation, with the added benefit of doing it silently. But at least you are forced to say doall
, which might make you feel at least a tiny bit guilty.
defn -main
("The entry point to your microservice."
[& args];; Do things...
println "I'm about to do...")
(
;; Spin wheels silently, until OOM or no more numbers,
;; whichever happens first.
doall (lazybuzz (range)))
(
;; Maybe not ...
println "I'm done!")) (
To see if you can get fired by solving fizzbuzz (now that's a concept, innit?), you can ship to production 9 the…
- doseq version, to fill up your log files with fizzbuzz. They will fill up really fast. Faster than logrotate.
- doall lazybuzz version, to confuse the daylights out of everyone, at least until your process dies with OOM.
- badly daemonised verison, to enjoy repeated restart cycles through crashes from number overflows and/or OOMs.
- Or something actually dangerous…
defn -main
("The entry point to your microservice."
[& args];; Do things...
println "I'm about to do...")
(
;; Your /thought/ you were going to /do/ something useful here.
range))
(fizz-buzz-classic (
;; You now falsely believe you did something useful ...
println "I'm done!")) (
I assure you, experienced Clojurists are no longer grinning at the tomfoolery. Many of us have shipped 10 (or almost shipped) this category of bugs to prod. Not fun.
OK, now I consider Le Cheval Classique suitably flogged postmortem, and yourself suitably Caveat Emptor-ed.
Now we will FizzBuzz joyously.
Little functions are good!
Once we remove the ick of println
from our code, we can see further room for improvement. (zero? (rem n divisor))
is not only a common pattern, it is actually a distinct idea, viz. "Is n
divisible by divisor
?".
We can name it locally, with let
.
defn letbuzz
(
[num-xs]for [n num-xs]
(let [divisible? (fn [n1 n2] (zero? (rem n1 n2)))] ; locally-bound lambda
(cond
(15) "FizzBuzz"
(divisible? n 3) "Fizz"
(divisible? n 5) "Buzz"
(divisible? n :else n))))
However, this definition of divisibility is generally applicable to numbers, so it makes sense to define a top-level global concept.
def divisible?
("True when the remainder of n1/n2 is zero. e.g. (divisible? 4 2) => true"
fn [n1 n2] (zero? (rem n1 n2)))) (
Yep, defn
is just def
+ fn
under the hood, and we can conveniently write the same thing as follows.
defn divisible?
("True when the remainder of n1/n2 is zero. e.g. (divisible? 4 2) => true"
[n1 n2]zero? (rem n1 n2))) (
We can also use comp
to define divisibility more succinctly.
def divisible?
("True when the remainder of n1/n2 is zero. e.g. (divisible? 4 2) => true"
comp zero? rem)) (
Since the various implementations of divisible?
are all pure functions, they are drop-in replacements for each other ("referentially transparent"). Use whichever version you like best.
It may seem silly to write such tiny functions, but we earn a lot of firepower by lifting out named domain concepts, especially the simple ones, because we can compose them flexibly to express other domain concepts as needed.
map reduce for FizzBuzz
Here's a doozy. By putting FizzBuzz logic inside for
, we have in fact deeply intertwined 11 two very distinct computations; viz. sequence generation, and choice-making.
Suppose we lifted out the decision logic into its own function?
defn basic-buzz
("We can also trivially rewrite this with 'condp'.
ref: https://clojuredocs.org/clojure.core/condp"
[n]cond
(15) "FizzBuzz"
(divisible? n 3) "Fizz"
(divisible? n 5) "Buzz"
(divisible? n :else n))
comment
(;; Unsurprisingly...
1) => 1
(basic-buzz 3) => "Fizz"
(basic-buzz 5) => "Buzz"
(basic-buzz 15) => "FizzBuzz"
(basic-buzz )
Now we can bring back for
this way…
def all-fizz-buzzes
(for [n (rest (range))]
( (basic-buzz n)))
But our new choice opens up the design space, because we can directly say "this is just a mapping of the domain of numbers to the domain of FizzBuzz".
def all-fizz-buzzes
(map basic-buzz (rest (range)))) (
Now since map
is conceptually just a special case of reduce
, we could write the following. However in Clojure, map
is lazy but reduce
is eager, and we only use reduce when we want to force a final calculation.
def just-some-fizz-buzzes
(reduce (fn [result n]
(conj result
(
(basic-buzz n)))
[];; the classic programmer's problem statement
range 1 101))) (
Once again, we earned more compositional power by lifting out another small concept. Let's do some more of that.
Domain Driven Design FizzBuzz
We can further define concepts specific to the business domain of FizzBuzz. This opens up our design space even more.
Before that I'll make one small tweak to help us express ourselves better. I'll rearrange the argument list of divisible?
so that the "more constant" argument is placed first, and successively more variable argument(s) are placed successively. Also rely on "truthiness" in Clojure to imply yes/no.
defn divisible?
("Given a number 'n', return the given word (truthy) when it is divisible
by the divisor, or nil otherwise (falsey)."
[divisor the-word n]when (zero? (rem n divisor))
(
the-word))
def fizzes?
("Is a given number divisible by 3?"
partial divisible? 3 "Fizz"))
(
def buzzes?
("Is a given number divisible by 5?"
partial divisible? 5 "Buzz"))
(
def fizzbuzzes?
("Is a given number divisible by 3 and 5?"
partial divisible? 15 "FizzBuzz")) (
Now we can rewrite basic-buzz
using or
, which short-circuits, and returns the first truthy value it encounters. You will see this construct in real-world Clojure code.
defn or-buzz
("Just like conditional matching, but exploit short-circuit behaviour of 'or'.
Sadly, order of conditionals still matters."
[n]or (fizzbuzzes? n)
(
(buzzes? n)
(fizzes? n) n))
We argued that we are essentially expressing a choice, and that we can even do it with juxt
, because once we grok juxt
, we want to use it everywhere.
defn juxt-buzz
("juxt for fun:
e.g. ((juxt f g h) 42) -> [(f 42) (g 42) (h 42)]
cf. https://clojuredocs.org/clojure.core/juxt
Sadly, order of conditional checks still matters, which combined with
the nil-punning that's going on here is too subtle for production use."
[n]some identity ((juxt fizzbuzzes? buzzes? fizzes? identity)
( n)))
Yeah, that's a head-scratcher. Best leave it back at home.
Actually Domain Driven FizzBuzz
You might protest that well actually the clever little functions, in fact, express the domain of the solution (the business of calculating FizzBuzz), not the domain of the problem (arithmetic representation of FizzBuzz).
And even though I flunked maths a lot, I would concur. So here goes nothing…
15 is the least common multiple of the prime factors. Suppose we cook up an encoding scheme based on remainders of 15, and write it down as a lookup table? We can then find (rem n 15)
, and look up the answer to FizzBuzz in the table.
Why do a lookup table? Well, what is the simplest possible function? A literal hard-coded lookup table!
In Clojure, we can use hash-maps to write down look-up tables.
;; A table of remainders of 15, in a hash-map.
0 "FizzBuzz"
{3 "Fizz"
6 "Fizz"
9 "Fizz"
12 "Fizz"
5 "Buzz"
10 "Buzz"}
And here is a very fun and useful fact. Clojure hash-maps are not just inert data. They also behave as functions of their keys. We can literally call ({:a 42} :a)
and get back 42. Noice!
So suppose we define a global lookup table?
def rem15->fizz-buzz
(0 "FizzBuzz"
{3 "Fizz"
6 "Fizz"
9 "Fizz"
12 "Fizz"
5 "Buzz"
10 "Buzz"})
comment
(rem 3 15)) => "Fizz"
(rem15->fizz-buzz (rem 5 15)) => "Buzz"
(rem15->fizz-buzz (rem 15 15)) => "FizzBuzz"
(rem15->fizz-buzz (rem 1 15)) => nil
(rem15->fizz-buzz ( )
See the nil
returned for "no result found"? If you were paying attention to the nil punning 12, and the short-circuiting or
, you might get the following idea. And you would not be wrong.
defn or-rem15-buzz
(
[n]or (rem15->fizz-buzz (rem n 15))
( n))
But we can be more right by using the get
function, which is designed for use with hash-maps, and which allows us to conveniently specify a fallback value for the "not found" case.
defn get-rem15-buzz
(
[n]get rem15->fizz-buzz
(rem n 15)
( n))
Not to press the point, but they are referentially transparent.
= (map or-rem15-buzz (range 1 16))
(map get-rem15-buzz (range 1 16))) (
You may have also astutely noted that, in both the implementations above, the order of calculation ceases to matter. Now we are doing maths.
FizzBuzz by construction
Closely related to remainder lookup tables, we can make the observation that FizzBuzz is cyclical in modulo 3, 5, and 15. So we can directly define the idea of FizzBuzz in those terms.
This FizzBuzz is correctly ordered by definition.
def mod-cycle-buzz
("We can declare a lazy sequence of FizzBuzz as modulo 3, 5, 15.
The sequence is ordered by definition."
let [n identity
(constantly "Fizz")
f (constantly "Buzz")
b (constantly "FizzBuzz")]
fb (cycle [n n f
(
n b f
n n f
b n f n n fb])))
Now, Clojure's map
is not only lazy, it can also apply a function of n
arguments over n
collections until any one of the collections is exhausted. So we can…
def all-fizz-buzzes
(map (fn [f n] (f n))
(; countless modulo pattern
mod-cycle-buzz rest (range)))) ; countless sequence of numbers (
If we think in terms of the prime factors 3 and 5, along with modulo cycles, it may inspire a generalised version of fizzbuzz, like so.
defn any-mod-cycle-buzz
("Given a number and a sequence of words mapping to prime factors,
either return the corresponding word-version for the number position,
or the number itself, if no prime factor exists.
Basically, the set of words should map to set of prime factors.
We also don't make any assumptions about order of words here. It is up
to the caller to choose whatever sequence they please."
num & words]
[or (not-empty (reduce str words))
(num))
map any-mod-cycle-buzz
(range 1 16)
(cycle [nil nil "Fizz"])
(cycle [nil nil nil nil "Buzz"])) (
And last but not least, this interpretation allows us to express the arithmetic identity of the whole category of FizzBuzzes, which is, just the number sequence itself. (As in, the identity of + is 0 and that of * is 1).
map any-mod-cycle-buzz
(range 1 16)) (
But then again, I've flunked maths too often to be confident about any of this. Please complain to me if I'm wrong.
Interlude: all the fizz-buzzes so far
I'll drop this mini pièce de résistance (for now), and pause for a breather. I've copied down all the fizz-buzz variants (minus doc strings for brevity).
ns all.them.fizz.buzzers)
(
def fizz-buzz map) ; now, what `map` can do fizz-buzz can too
(
;; Le FizzBuzz Classique Functional
defn basic-buzz
(
[n]let [divisible? (comp zero? rem)]
(cond
(15) "FizzBuzz"
(divisible? n 5) "Buzz"
(divisible? n 3) "Fizz"
(divisible? n :else n)))
;; Branching logic FizzBuzzes
defn divisible?
(
[divisor the-word n]when (zero? (rem n divisor))
(
the-word))
def fizzes? (partial divisible? 3 "Fizz"))
(def buzzes? (partial divisible? 5 "Buzz"))
(def fizzbuzzes? (partial divisible? 15 "FizzBuzz"))
(
defn or-buzz
(
[n]or (fizzbuzzes? n)
(
(buzzes? n)
(fizzes? n)
n))
defn juxt-buzz
(
[n]some identity
(juxt fizzbuzzes? buzzes? fizzes? identity)
((
n)))
;; More mathematical FizzBuzzes
def rem15->fizz-buzz
(0 "FizzBuzz"
{3 "Fizz"
6 "Fizz"
9 "Fizz"
12 "Fizz"
5 "Buzz"
10 "Buzz"})
defn or-rem15-buzz
(
[n]or (rem15->fizz-buzz (rem n 15))
(
n))
defn get-rem15-buzz
(
[n]get rem15->fizz-buzz
(rem n 15)
(
n))
def mod-cycle-buzz
(let [n identity
(constantly "Fizz")
f (constantly "Buzz")
b (constantly "FizzBuzz")]
fb (cycle [n n f
(
n b f
n n f
b n f
n n fb])))
defn any-mod-cycle-buzz
(num & words]
[or (not-empty (reduce str words))
(num))
;; Inspect and check the fizz-buzzes
defn call-all-fizz-buzzers
(
[range-of-buzzees]
[(fizz-buzz basic-buzz
range-of-buzzees)
(fizz-buzz or-buzz
range-of-buzzees)
(fizz-buzz juxt-buzz
range-of-buzzees)
(fizz-buzz or-rem15-buzz
range-of-buzzees)
(fizz-buzz get-rem15-buzz
range-of-buzzees)fn [f n] (f n))
(fizz-buzz (
mod-cycle-buzz
range-of-buzzees)
(fizz-buzz any-mod-cycle-buzz
range-of-buzzeescycle [nil nil "Fizz"])
(cycle [nil nil nil nil "Buzz"]))])
(
defn check-all-fizz-buzzers
("Return true if all known fizz-buzzers produce equal results
for the programmer's FizzBuzz for numbers 1 to 100"
[]apply = (call-all-fizz-buzzers (range 1 101)))) (
And lest we forget, let us flog the dead horse one last time.
defn severely-broken-buzz
("Please don't do this in Clojure."
[n]cond
(15) (println "FizzBuzz")
(divisible? n 3) (println "Fizz")
(divisible? n 5) (println "Buzz")
(divisible? n :else (println n)))
The mind is abuzz drafting moaaaar variants… Stay tuned!
Peano FizzBuzz
Since we are computing with natural numbers, we can express FizzBuzz in terms of the "Successor" operation of Peano arithmetic.
However, we have to modify our number system a bit to make it work right. We define a PeanoBuzz number to be a pair of a natural number and its FizzBuzz counterpart. The PeanoBuzz number system starts at [0 0]
.
We can enjoy the fruits of compositionality that we planted earlier.
ns all.them.fizz.buzzers)
(
def rem15->fizz-buzz
(0 "FizzBuzz"
{3 "Fizz"
6 "Fizz"
9 "Fizz"
12 "Fizz"
5 "Buzz"
10 "Buzz"})
defn get-rem15-buzz
(
[n]get rem15->fizz-buzz
(rem n 15)
(
n))
def S
("The PeanoBuzz number system starting at [0 0] is closed under
this definition of Successor."
comp (juxt identity get-rem15-buzz)
(inc
first))
def all-peano-buzzes
(iterate S [0 0]))
(
comment
(= (take 16 all-peano-buzzes)
(0 0] [1 1] [2 2]
[[3 "Fizz"] [4 4]
[5 "Buzz"]
[6 "Fizz"] [7 7] [8 8]
[9 "Fizz"]
[10 "Buzz"] [11 11]
[12 "Fizz"] [13 13] [14 14]
[15 "FizzBuzz"]])) [
Since we have a S
that closes over the PeanoBuzz number system, I wonder if we can satisfy all the Peano Axioms? Probably another blog post :)
Incidentally, we can trivially map the domain of PeanoBuzz back into the domain of FizzBuzz.
ns all.them.fizz.buzzers)
(
= (fizz-buzz basic-buzz
(range 1 101))
(second
(fizz-buzz take 100 (rest all-peano-buzzes)))) (
We are already half way to Lambda Calculus / Church Numerals. But going there will side-track our FizzBuzz expedition way too much. So I'll leave the Church Numerals version as an exercise to the reader 13. Try to make it so that that we can define an all-church-nums-buzz
and slot it into the fizz-buzz
checks we already have.
Dispatch Buzz
If you squint at the conditional FizzBuzzes, like basic-buzz
, or-buzz
etc., you might re-see them as a dispatch problem. And why would you be wrong? They, like any other if-else-y construct are truth tables hard-wired to "finalised" values or operations.
Naturally we will follow the consummate Clojurian's primal instinct of needing to pry apart two things masquerading as one ("decomplecting" 14 in Clojurish). But, what exactly are we, ah, decomplecting? Now that is a very interesting rabbit hole.
In this case we are "separating mechanism from policy" 15. Figuring out how to do this delivers a powerful, flexible method of program design into our eager hands.
This table shows "mechanism" and "policy" hard-wired together.
<-- ------- MECHANISM -------- -->|<-- POLICY -->
| n divisible? 3 | n divisible? 5 | Final value |
|----------------+----------------+-------------|
| true | true | FizzBuzz |
| true | false | Fizz |
| false | true | Buzz | | false | false | n |
Here is an attempt to pry the two apart.
Mechanism
The "mechanism" here is any function that translates a number (or more generally, any thing) to the two inputs of a 2-value truth table. We can see it more clearly if we rewrite the input columns of the truth table like this. Here f
and g
are functions of a
to Boolean.
| (f a) | (g a) |
|-------+-------|
| true | true |
| true | false |
| false | true | | false | false |
We can express this as a Clojure punction 'coz (f a) (g a) is ((juxt f g) a).
ns dispatch.buzz)
(
defn mechanism
("Given two functions, presumably of anything -to-> Boolean, return
a function that can construct inputs of a 2-input truth table."
[f? g?]juxt f? g?)) (
See? Such abstract. Much general-purpose. Very decomplect. Wow.
Policy
Here, we define "policy" as something having special context of FizzBuzz that consumes input rows of the truth table and emits corresponding output fields.
First, we specialise the abstract mechanism
to a FizzBuzz mechanism. You may smirk 16, but just you wait. There is a (multi) method to the madness…
Here is the table we started with, rewritten for our specialisation.
| (fizzes? n) | (buzzes? n) | (mechanism fizzes? buzzes?) -> mfb |
|-------------+-------------+------------------------------------|
| true | true | (mfb n) => "FizzBuzz" |
| true | false | (mfb n) => "Fizz" |
| false | true | (mfb n) => "Buzz" | | false | false | (mfb n) => n |
Here is the intermediate step of the specialisation, viz. (mechanism f g) -> h
.
ns dispatch.buzz)
(
defn divisible?
(
[divisor n]zero? (rem n divisor)))
(
def fizzes? (partial divisible? 3))
(def buzzes? (partial divisible? 5))
(
;; If we take the abstract mechanism and give it functions
;; that test numbers for fizzery and buzzery, then we can
;; construct a version of the truth table that is /concretely/
;; specific to FizzBuzz.
comment
(= (map (mechanism fizzes? buzzes?)
(15 3 5 1])
[true true]
[[true false]
[false true]
[false false]])
[ )
Finally, we hang it all together with Clojure's multimethods 17, like so.
ns dispatch.buzz)
(
def fizz-buzz map)
(
def fizz-buzz-mechanism
(
(mechanism fizzes? buzzes?))
defmulti dispatch-buzz
("It yields the third column of the truth table."
fizz-buzz-mechanism)
;; The /Policy/, fully realised.
;; ((mechanism fizzes? buzzes?) n) -> final results
defmethod dispatch-buzz [true true]
(
[n]"FizzBuzz")
defmethod dispatch-buzz [true false]
(
[n]"Fizz")
defmethod dispatch-buzz [false true]
(
[n]"Buzz")
defmethod dispatch-buzz :default
(
[n]
n)
;; The /Policy/, applied.
comment
(= (fizz-buzz dispatch-buzz
(1 3 5 15 16])
[1 "Fizz" "Buzz" "FizzBuzz" 16])
[ )
Yes, 'tis a wee FizzBuzz interpreter!
(Pirouettes off-stage, gracefully.)
Embarrassingly Parallel FizzBuzz
It turns out FizzBuzz is one of those so-called Embarrassingly Parallel problems.
ns all.them.fizz.buzzers)
(
def fizz-buzz map)
(
;; Add 1 character for parallel map.
def embarrassingly-parallel-fizz-buzz pmap)
(
comment
(let [range-of-buzzees (range 1 101)]
(= (fizz-buzz basic-buzz
(
range-of-buzzees)
(embarrassingly-parallel-fizz-buzz basic-buzz
range-of-buzzees))) )
I was almost too embarrassed to write this, but I'm glad good sense prevailed, because the trivial replacement of map
with pmap
teaches a lesson.
Parallelism is impossibly hard if we don't have pure functions, immutability, and laziness. When we do, it reduces to merely hard but tractable. The proverbial single character code modification (in this case, literally so) gets a free ride on those other constructs.
We can see it in pmap
's implementation. It is fairly straightforward. Fetch its source and stare at it for a bit; (clojure.repl/source pmap)
. You will be able to make sense of it with a bit of cross-referencing ClojureDocs.
If your favourite language has something similar (parallel version of a sequential function), and if you choose to compare implementations, I will be delighted to learn from your analysis!
OOP Buzz
What is an Object in, say, Java? It is a combination of four distinct things:
- Name (Class name / Java type)
- Structure (Class members, methods etc.)
- Behaviour (effects caused by methods)
- State (contained in the run-time instance of the Class)
In Clojure, all of these are separate ("available a la carte", in Clojurish), the consequences of which are hard to explain sans a motivating example.
Clojure's approach to Polymorphism allow us to do things like this.
ns oops.fizzbuzz)
(
def divisible? (comp zero? rem))
(def fizz-buzz map)
(
;; Like a Java Interface, but better.
defprotocol IFizzBuzz
(
(proto-buzz [this]))
;; We can add new behaviour to existing types,
;; including /any/ Java built-in type.
extend-protocol IFizzBuzz
(
java.lang.Number
(proto-buzz [this]cond
(15) "FizzBuzz"
(divisible? this 3) "Fizz"
(divisible? this 5) "Buzz"
(divisible? this :else this)))
comment
(;; Now we can do this
= (fizz-buzz proto-buzz
(1 3 5 15 16])
[1 "Fizz" "Buzz" "FizzBuzz" 16])
[
;; And we can also do this
= (fizz-buzz proto-buzz
(1.0 3.0 5.0 15.0 15.9])
[1.0 "Fizz" "Buzz" "FizzBuzz" 15.9])
[
;; WITHOUT breaking any existing Equality semantics.
;; (= 42 42) => true (long and long)
;; (= 42 42.0) => false (long and double)
;; (= 42.0 42.0) => true (double and double)
;; Thus, this is false, as it should be.
= (fizz-buzz proto-buzz
(1 3 5 15 16])
[
(fizz-buzz proto-buzz1.0 3.0 5.0 15.0 15.9]))
[ )
Not to put too fine a point on it, but Clojure is a far more capable Object Oriented Programming System than say Java or Kotlin, a fact which I have personally profited from handsomely in the past 18.
Why?
In short, Clojure cleanly solves the "Expression Problem".
In long, you can…
- Watch Clojure Protocols Explained, by Sean Devlin.
- Watch Clojure's Solutions to the Expression Problem, by Chris Houser.
- Listen to Rich Hickey on Protocols and Clojure 1.3, by Rich Hickey.
- Read "Rich Already Answered That!", curated by reborg. "A list of commonly asked questions, design decisions, reasons why Clojure is the way it is as they were answered directly by Rich."
- Read the parable of The venerable master Qc Na and his student Anton. This HN thread also has some interesting links (and diatribes).
In very short… Tweet this, look smart, get paid!
If you do Functional Programming right, you get Object Oriented Programming for free #Clojure. (And vice-versa e.g. #Erlang #Smalltalk #OCaml).
- Yours Truly
Non-Destructive FizzBuzz
proto-buzz
is a great motivating example of what I would like to call the Non-Destructive FizzBuzz.
All the FizzBuzz solutions seen previously, except PeanoBuzz, lose information. This is almost always bad because its impossible to reverse information loss. The inverse is also true. It is almost always good to augment information. Prefer to enrich information and retain as much as resources permit (organised neatly, of course). Ask any lawyer or accountant or friendly neighbourhood Clojurian which alternative would set their hair on fire.
Here I play with some more ways to FizzBuzz non-destructively.
As I do, I meditate upon the extra super nice thing about proto-buzz
. Which is that we did not have to invent a new number system or box numbers in some composite FizzBuzz data representation and we lost no functionality — numbers still behave exactly as we expect, with zero added overhead!
Composite Data Buzz
PeanoBuzz was an example of writing FizzBuzz in terms of "composite" data. Abstractly, that idea is basically "attach some meta-data to my things".
PeanoBuzz was a tuple [3 "Fizz"]
, but it could very well have been a custom map representation, say, {:n 3 :fizzbuzz "Fizz"}
.
We can do one better by using Clojure Records to attach full-blooded Java types to our numbers 19 and also make "composite" data, because Records give us all the features of generic hash-maps for free.
ns boxed.fizz.buzz)
(
def divisible? (comp zero? rem))
(def fizz-buzz map)
(
defrecord Fizz [n])
(defrecord Buzz [n])
(defrecord FizzBuzz [n])
(defrecord Identity [n])
(
defn boxed-buzz
(
[n]cond
(15) (->FizzBuzz n)
(divisible? n 3) (->Fizz n)
(divisible? n 5) (->Buzz n)
(divisible? n :else (->Identity n)))
def all-boxed-buzzes
(map boxed-buzz
(rest (range))))
(
comment
(;; Composite hash-map-like data!
= (fizz-buzz boxed-buzz
(1 3 5 15])
[:n 1}
[#boxed.fizz.buzz.Identity{:n 3}
#boxed.fizz.buzz.Fizz{:n 5}
#boxed.fizz.buzz.Buzz{:n 15}]) ; => true
#boxed.fizz.buzz.FizzBuzz{
;; Which is nondestructive!
= [1 3 5 15]
(comp :n boxed-buzz)
(fizz-buzz (1 3 5 15])) ; => true
[
;; And which has real Java types!
= (map type (fizz-buzz boxed-buzz [1 3 5 15]))
(
[boxed.fizz.buzz.Identity
boxed.fizz.buzz.Fizz
boxed.fizz.buzz.Buzz; => true
boxed.fizz.buzz.FizzBuzz]) )
Further exercises for the dear reader!
- Re-implement PeanoBuzz using Clojure hash-maps!
- Re-re-implement PeanoBuzz with Records!
- Separately write something that can return the classic string-or-number equivalent of your composite types! (Hint: use multimethods and/or protocols as appropriate).
Last but not least, ask yourself "But what about equality? And the rest of arithmetic?" while comparing these with proto-buzz
.
Clojure Spec'd Buzz
Off-label use of Clojure Spec's conform
as a parser can be very handy.
ns conformer.buzz)
(require '[clojure.spec.alpha :as s])
(
defn divisible?
(
[divisor n]zero? (rem n divisor)))
(
def fizzes? (partial divisible? 3))
(def buzzes? (partial divisible? 5))
(
::number number?)
(s/def ::fizzes (s/and ::number fizzes?))
(s/def ::buzzes (s/and ::number buzzes?))
(s/def
comment
(;; Now we can parse input data...
::fizzes 3) ; => 3
(s/conform ::buzzes 5) ; => 5
(s/conform ::buzzes 3) ; => :clojure.spec.alpha/invalid
(s/conform ::fizzes ::buzzes) 15) ; => 15
(s/conform (s/and
;; And we can handle non-conforming data gracefully,
;; instead of panicking and throwing exceptions.
::fizzes ::buzzes) "lol") ; :clojure.spec.alpha/invalid
(s/conform (s/or
)
def fizz-buzz-specs
("Set of FizzBuzz parsers."
::fizzes ::buzzes ::number})
#{
defn spec-parse-buzz
("Conform the given input to the set of specs for fizz-buzz, and return
a pair of the input and a map of parsed values keyed by the parser name."
[x]zipmap fizz-buzz-specs
[x (map #(s/conform % x) fizz-buzz-specs))])
(
def all-spec-buzzes
(map spec-parse-buzz
(rest (range))))
(
comment
(;; And we can...
take 100 all-spec-buzzes)
(
;; Which gives us enriched data, like this:
= (into {} (map spec-parse-buzz [1 3 5 15 "lol"]))
(1 #:conformer.buzz{:fizzes :clojure.spec.alpha/invalid,
{:buzzes :clojure.spec.alpha/invalid,
:number 1},
3 #:conformer.buzz{:fizzes 3,
:buzzes :clojure.spec.alpha/invalid,
:number 3},
5 #:conformer.buzz{:fizzes :clojure.spec.alpha/invalid,
:buzzes 5,
:number 5},
15 #:conformer.buzz{:fizzes 15,
:buzzes 15,
:number 15},
"lol" #:conformer.buzz{:fizzes :clojure.spec.alpha/invalid,
:buzzes :clojure.spec.alpha/invalid,
:number :clojure.spec.alpha/invalid}}) ; => true
)
However, like off-label use of anything, this conform
-as-parser trick too skirts the "can be a very bad idea" territory.
YMMV.
Wicked pprint Buzz
@rdivyanshu said to add this extra pprint dispatcher, "and let no number escape fizzbuzzness when showing itself".
Why not?
ns pprint.buzz)
(require '[clojure.pprint :as pp])
(
defn pprint-buzz
(
[n]let [divisible? (comp zero? rem)
(comp prn
prettyprint (partial format "%d doth %s"))]
(cond
(15) (prettyprint n "FizzBuzzeth")
(divisible? n 3) (prettyprint n "Fizzeth")
(divisible? n 5) (prettyprint n "Buzzeth")
(divisible? n :else (prettyprint n "not Fizzeth nor Buzzeth. Alas!"))))
comment
(;; lol
#'pp/use-method pp/simple-dispatch java.lang.Number pprint-buzz)
(
;; nothing to see here... you will have to look at the REPL
doseq [n [1 3 5 15]]
(;; lol lol lol lolllll
(pp/pprint n)) )
You, Sir, are truly a gentlenerd and a scholar.
Nondestructive, and hilarious to boot!
Tagged Literal Buzz
Thinking aloud…
Clojure has a concept of "tagged literals"; plaintext labels that we can "attach" to data without changing the literal value of our data, and transmit over wires along with the data.
Clojure provides built-in support for a small set of some fairly universal types of literals (instant, UUID etc.). And I have used fancier tagged literals, but only in context of other people's libraries (juxt/aero).
How to make this work, I wonder?
ns tagged.buzz)
(
;; We can do this, but how to work with it?
(set! *default-data-reader-fn* tagged-literal)
def tagged-buzz
(3
#tagged.buzz/FizzBuzzSequence [#tagged.buzz/Fizz 5
#tagged.buzz/Buzz 15
#tagged.buzz/FizzBuzz 1]) #tagged.buzz/Number
Note: Parked for now. I'm not sure if this a reasonably real-world FizzBuzz.
Interlude: What more could we possibly decomplect?
Well, in one word, Transducers.
Which you can assume is Clojurish for "You haven't decomplected your Clojure quite enough just yet.". But before we go there, a quick survey.
Thus far, we have pulled apart the FizzBuzz problem space in many dimensions.
- Calculation v/s Effects (lifted out
println
) - Calculation v/s Sequence generation (lifted fizzbuzz logic out of
for
) - Definition v/s Realisation (lazy definitions like
all-fizz-buzzes
) - Lifted out concepts in the domain of the solution (
fizzes?
buzzes?
etc.) - Lifted out concepts in the domain of the problem (modulo math FizzBuzzes)
- Lifted all fizzbuzzes out of the concrete 3/5 FizzBuzz (
any-mod-cycle-buzz
) - Added Fizzbuzz-meaning to Numbers without changing existing Number-meaning (protocols, and maybe also tagged literals if I can make it work sensibly)
- Concrete Numbers v/s abstract representations (PeanoBuzz,
mod-cycle-buzz
) - Calculation v/s Specification (
spec-parse-buzz
) - We even teased apart printing context (wicked
pprint-buzz
)
Note that all of this belongs in the real-world Clojurian's design toolbox.
Not only do we do it "in the small", in our little namespaces and monoliths polyliths, we also do it "in the large" in our data-center-spanning distributed systems.
But you see, in all the teasing apart so far, we implied FizzBuzz was in-memory sequence processing. What if we could not make any assumption whatsoever about data source, or data size, or data format, or process control, or the output sink? Can we still describe FizzBuzz in some useful way?
Well, in one word, Transducers.
Savvy Clojurians may appreciate that text above has transducer signature.
(whatever, input -> whatever) -> (whatever, input -> whatever)
"Seems like a good project for the bar, later on."
- Rich Hickey
Transducery Buzz
This is really my feeble attempt to nerdsnipe you into falling into deep abstraction territory.
I pray that you give yourself time to absorb transducers. Peruse the code below. Then peruse the list of stimulating material that follows, over a relaxed weekend with several cups of delicious artisanal Oolong to salve the brain pain.
What are we decomplecting?
Because, with transducers, we will now also pull apart:
- Data source (sequence, stream, channel, socket etc.)
- Data sink (sequence, stream, channel, socket etc.)
- Data transformer (function of any value -> any other value)
- Data transformation process (mapping, filtering, reducing etc.)
- Some process control (we can transduce finite data (of course) as well as streams, and also have optional early termination in either case. I'm not sure about first-class support for other methods like backpressure.)
Of course, for useful computations, we have to compose these back in some sensible way, appropriate to the context at hand (e.g. read numbers off a Kafka topic, FizzBuzz them, and send them to another Kafka topic, OR slurp numbers from file on disk, FizzBuzz them, and push into an in-memory queue).
A word of caution. As you read, you may think "Ew, he peddles oldass Unix Pipes." 20 (or shiny new Monads or Lambda Architecture(TM) or some familiar-to-you generics). You won't be wrong. You won't be right either.
Because, as is true for all sufficiently abstract abstractions, analogies are not equivalences. Details differ deeply and your brain will hurt at first. A lot. For example, it is not obvious why, but the transducer's mandate of a la carte re-composition demands that all the new pulling apart must be fully compatible with all the old pulling apart.
'nuff said. Decomplecting our Clojure 'smore in 3… 2… 1…
ns transducery.buzz)
(
def divisible?
(comp zero? rem))
(
defn basic-buzz
(
[n]cond
(15) "FizzBuzz"
(divisible? n 3) "Fizz"
(divisible? n 5) "Buzz"
(divisible? n :else n))
zzzzz. snort. Old news. Whatever.
Demo One: Computation and Output format pulled apart
ns transducery.buzz)
(
;; Separately define /only/ the transformation "xform"
def fizz-buzz-xform
(comp (map basic-buzz)
(take 100))) ;; early termination
(
;; Separately define /only/ input data
def natural-nums
(rest (range)))
(
;; Compose to produce a sequence
comment
(;; calculate each step
(transduce fizz-buzz-xform conj ;; and use this output method
;; to pour output into this data structure
[] ;; given this input
natural-nums)
)
;; Compose differently to produce a string
comment
(;; calculate each step
(transduce fizz-buzz-xform str ;; and use this output method
"" ;; to catenate output into this string
;; given this input
natural-nums)
)
;; Compose still differently to produce a CSV string
defn suffix-comma
(
[s]str s ","))
(
comment
(comp fizz-buzz-xform
(transduce (map suffix-comma)) ;; calculate each step
(str ;; and use this output method
"" ;; to catenate output into this string
;; given this input
natural-nums)
)
Pause for a bit.
- Consider the parts we did not have to modify at all even though we modified everything about the output and about the xform.
- Consider what it might take to reuse any of the other fizzbuzzers instead of
basic-buzz
. - Try it!
Demo Two: Computation and Input format pulled apart.
ns transducery.buzz)
(
;; Setup
def numbers-file
("Plaintext file containing numbers in some format."
"/tmp/deleteme-spat-by-clj-fizz-buzz-demo.txt")
;; Write 10,000 natural numbers to file, one per line
#_(spit numbers-file
"\n" (range 1 10001)))
(clojure.string/join
;; Read back to double-check we got it.
#_(slurp numbers-file)
;; For contrast: This is how we might fizz-buzz traditionally.
comment
(;; Like this, if we don't know our threading macros.
;; (Don't fret about it. This is just fine.)
let [fizz-buzz (fn [s] (basic-buzz (Integer/parseInt s)))]
(take 15
(map fizz-buzz
(slurp numbers-file)))))
(clojure.string/split-lines (
;; Or more Clojurishly, with our nifty threading macros.
->> numbers-file
(slurp
clojure.string/split-linesmap #(basic-buzz (Integer/parseInt %)))
(take 15))
(;; I interrupted us with this, because of a pet peeve. People like to
;; describe this form as a "pipeline". It isn't. It is a formatting
;; sleight of hand that makes in-process call stacks of functions
;; /appear/ to be straight-line. The resulting shape visually suggests
;; punching data through a pipeline.
;;
;; Whereas pipelines are fundamentally streaming abstractions that
;; cross process boundaries.
;;
;; Transducers + xforms are highly pipeline-like.
)
;; Apart from not really being pipelines, both these traditional versions
;; are also hopelessly complected with sequences, which malady is addressed
;; by this transducer version.
comment
(comp (map #(Integer/parseInt %))
(transduce (;; calculate each step
fizz-buzz-xform) conj ;; and use this output method
;; to pour output into this data structure
[]
(clojure.string/split-linesslurp numbers-file))) ;; given this input
( )
A reader may complain that split-lines and file slurpin' is still complected. The reader would be right. Tim Baldridge's video series listed below will help work out how one might go about transducing over numbers-file directly.
Demo Three: Use only the xform as a calculator
ns transducery.buzz)
(
;; The xform can still calculate just a single item:
conj) [] 3) ;; => ["Fizz"]
((fizz-buzz-xform str) "" 3) ;; => "Fizz"
((fizz-buzz-xform str) "" 1) ;; => "1"
((fizz-buzz-xform fn [_ out] out)) nil 3) ;; "Fizz"
((fizz-buzz-xform (fn [_ out] out)) nil 1) ;; 1 ((fizz-buzz-xform (
Hopefully now it is a little more obvious why the transducer's mandate of a la carte re-composition demands that all the new pulling apart must be fully compatible with all the old pulling apart.
Further reading
Transducers are very deep conceptually, and literally. Since Clojure 1.7, they have come to underpin all of Clojure's heavy-lift capability.
I recommend drilling down this way.
Thirty minute quickstart
- Skim-read the official words introducing Transducers, and describing "What are good use cases for transducers?".
- Watch Tim Baldridge lift the essence of transducers out from map/filter/reduce in 10 minutes: Transducers - Episode 1 - Introduction to Transducers
- Compare with
(clojure.repl/source map)
(and filter and reduce).
Half day binge watch
- Watch Rich Hickey introduce Transducers
- Follow Tim Baldridge through 9 short video demos, where he "draws the rest of the Owl" so to speak, but actually, with all the intermediate steps accounted for. IMHO, this is hands-down the best exploration of transducers out there 21.
- Watch Rich Hickey dive Inside Transducers.
Grok some real-world Transduction
- Kafkaesquely Streaming Transducery
- Grammar Transduction
- Transducers Haskellized
- Structure and Interpretation of Clojure Transducers re:Clojure 2021 workshop, by Ben Sless
Exercise your brain
- Write a FizzBuzz in Shell that can compute with any source/sink combination; in-line seq, mkfifo, files, sockets, URLs.
- Next, replace only the Shell FizzBuzz function with the
basic-buzz
function we wrote (use babashka). - Finally, write an all-Clojure version around the
basic-buzz
function, without losing the the ability to transparently read/write from/to seq, mkfifo, file, socket, URL.
man bash |
tr -cs A-Za-z '\n' |
tr A-Z a-z |
sort | uniq -c | sort -rn |
sed 10q
TODO Buzz
Ideas on deck, to put self on the hook…
Outside of clojure.core, maaaaybe core.async fizzbuzz, but IDK, maybe that will be too high concept, and too contrived.
Acknowledgments
Thanks to @rdivyanshu for review and feedback and ideas.