Threading Macros In Clojure 2: Electric Boogaloo

Earlier I wrote about Clojure’s humble threading macro. It turns out that like Transformers and ant nests there is more to threading macros in Clojure than meets the eye. Also, they are apparently called thrush operators.

->>, or Thread Last

-> threads an expression as the second item in an function call. In comparison, ->> threads the expression as the last item. Think of the extra angle bracket as a speed line showing how the arrow is going faster, so it flies in to the last position of the function call.

This is particularly helpful when working with sequences and functions that operate on sequences since they usually take a function as the second argument and the sequence as the last argument. For example, lets say we want to yell about what things we don’t like to see in job postings.

(def bad-signs-in-job-postings ["facebook for" "rockstar" "guru" "ninja" "bro" "red bull" "crush" "disrupt"])
; actually disrupt is ok
(apply .toUpperCase (apply str (filter #((not (= "disrupt" %))) (shuffle bad-signs-in-job-postings))))

This looks a little tricky. Since apply, filter, and most of the functions that operate on seqs take the sequence in the last position, -> won’t help us here. What is that whooshing sound you hear? It is ->>, whizzing in to the rescue! Let’s rewrite it and see how it looks.

(def bad-signs-in-job-postings ["facebook for" "rockstar" "guru" "ninja" "bro" "red bull" "crush" "disrupt"])
; actually disrupt is ok
(->> bad-signs-in-job-postings shuffle (filter #((not (= "disrupt" %)))) (apply .toUpperCase) (clojure.string/join " "))

We thread bad-signs-in-job-postings through shuffle, filter, apply, and join, with the result of each function call getting inserted in the last position of the next function call. Now we are yelling about dumb startup rhetoric! Neat!

as->, or Thread However I Want

What if you want to mix up where something gets threaded? You could nest -> and ->>, but that can get sticky icky quickly. Fortunately, Clojure 1.5 added as->, which lets you go wild with threading.

as-> takes an expression, a name for that expression, and a bunch of forms that you use that name in. In each form the name represents the result of evaluating the previous form. Prose is hard, let’s go coding.

(as-> [1 2 3 4] $
  (seq $)
  (nth $ 0)
  (+ $ 5)
; 6

Coming up with a non-contrived example is left as an exercise to the reader.

Other Threading Macros

There are a few other specialized threading macros. cond-> and it’s sibling cond->> allow you to branch before threading, and some-> and some->> will thread a result through as long as it is not nil. My brain is tired so I failed to write examples of using these, but it is helpful to know they are out there. Now go forth and thread!