Transducers

 

Before We Start

:dependencies [[org.clojure/clojure "1.7.0-alpha1"]
               [org.clojure/core.async "0.1.338.0-5c5012-alpha"]]

Reduce

https://clojure.github.io/clojure/clojure.core-api.html#clojure.core/reduce

Reduce

function

Usage: (reduce f col)

             (reduce f val col)

Overly simple example:

(reduce + [1 2 3 4 5])

Reducing Function

This time less trivially:

(defn word-counts 
  [running-counts next-word]
  (update-in running-counts [next-word] 
    (fnil inc 0)))

user=> (word-counts {"a" 1} "a")
{"a" 2}

user=> (word-counts {"a" 1} "b")
{"b" 1, "a" 1}

user=> (reduce word-counts {} (str/split "that man is not that great , man" #" "))
{"," 1, "great" 1, "not" 1, "is" 1, "man" 2, "that" 2}

A reducing function takes the prior reduction, the next element, and returns a new redution

More Reducing Functions

(defn word-counts 
  [running-counts next-word]
  (update-in running-counts [next-word] 
    (fnil inc 0)))

Rich calls this fn signature
  whatever, input -> whatever

But this doesn't tell the full story, because whatever in the input can be a totally different thing than the whatever in the output.

Reducing FNs Are General

CAN EXPRESS ALL SORTS OF THINGS LIKE MAP AND FILTER

(defn myfilter 
  [predicate] 
  (fn [running next] 
    (if (predicate next) 
      (conj running next) 
      running)))

user=> (reduce (myfilter #(= 2 %)) [] [1 2 3 4 3 2 1])
[2 2]

Write map as an exercise.

But At The End of the Day, We're coupled!

(defn myfilter 
  [predicate] 
  (fn [running next] 
    (if (predicate next) 
      (conj running next) 
      running)))

 

Do We Have To Be?

(defn myfilter2
  [predicate] 
  (fn [reducing]
    (fn [running next] 
      (if (predicate next) 
        (reducing running next) 
        running))))

Back To Rich's function signatures:

A *transducer* is 
  reducing fn -> reducing fn
Or as he puts it
  (whatever, input -> whatever) -> (whatever, input -> whatever)

Simple Transducer Example

This satisfies the function signature Rich gave.

(defn lower-words 
  [reducing]
  (fn [whatever input]
    (reducing whatever (.toLowerCase input))))

(require '[clojure.string :as str])

user=> (reduce (lower-words word-counts) {} 
         (str/split "That man is not that great , man" #" "))
{"," 1, "great" 1, "not" 1, "is" 1, "man" 2, "that" 2}

Wait If these All Share a signature...

Then yes! We can comp them!

(reduce 
  ((comp
     lower-words (myfilter2 #(= "that" %))) 
    word-counts)
  {} (str/split "That man is not that great , man" #" "))


{"that" 2}

Are We There?

Yes.

 

Also note that we aren't coupled to a data type, that's in the reducing fn.  So we can generally apply this to all kinds of stuff.

Now With Transduce

(transduce (map #(.toLowerCase %)) 
           word-counts {} 
           (str/split "That man is not that great , man" #" "))
(time 
  (doseq [x (range 1000)] 
    (->> (str/split "That man is not that great , man" #" ") 
         (map #(.toLowerCase %))
         (reduce word-counts {}))))
"Elapsed time: 57.015 msecs"

(time 
  (doseq [x (range 1000)] 
    (transduce (map #(.toLowerCase %)) 
               word-counts {} 
               (str/split "That man is not that great , man" #" "))))

"Elapsed time: 47.94 msecs"

Also a performance gain (maybe)

Also works with core.async

(require '[clojure.core.async :as async])

(let [channel (async/chan 2 (map #(.toLowerCase %)))] 
  (async/go 
    (async/>! channel "Hello World!") 
    (prn (async/<! channel))))

#<ManyToManyChannel clojure.core.async.impl.channels.ManyToManyChannel@75acd47>
"hello world!"

You TRy One...

Write FizzBuzz with transducers.  This is not exactly an intuitive fit, but you can write a FizzBuzz, a Fizz, and a Buzz, and cond them for this.

 

 

 

Transducers

By Philip Doctor

Transducers

  • 1,798