I have a calculation someting like the following:
;; for sake of simplicity we use round numbers
(def data [{:a 1} {:a 10} {:a 100}])
(reduce - 0.0 (map :a data))
And it evaluates to -111.0
. I want to do the transformation with a transducer to speed it up a bit by preventing unnecessary allocations:
(transduce (map :a) - 0.0 data)
However, the signum of the result changed to positive! Apparently it does not matter if I use +
or -
as the reducer in the expression, as the form will evaluate to +111.0
in both cases.
This is surprising to me, why did the introduction of transduce
change the semantics, what am I missing here?
(the strange behaviour happens with *
and /
too!)
So apparently
transduce
calls the reducer function on the result of the reduction, so-
just flips the sign when the reduction is done. We need to wrap the reducer function with completing to hide this call:Although, at this point it is shorter to just write: