I’m having some trouble understanding how the delay macro works in Clojure. It doesn’t seem to do what expect it to do (that is: delaying evaluation). As you can see in this code sample:
; returns the current time
(defn get-timestamp [] (System/currentTimeMillis))
; var should contain the current timestamp after calling "force"
(def current-time (delay (get-timestamp)))
However, calling current-time in the REPL appears to immediately evaluate the expression, even without having used the force macro:
user=> current-time
#<Delay@19b5217: 1276376485859>
user=> (force current-time)
1276376485859
Why was the evaluation of get-timestamp not delayed until the first force call?
The printed representation of various objects which appears at the REPL is the product of a multimethod called
print-method. It resides in the filecore_print.cljin Clojure’s sources, which constitutes part of what goes in theclojure.corenamespace.The problem here is that for objects implementing
clojure.lang.IDeref— the Java interface for thingsderef/@can operate on —print-methodincludes the value behind the object in the printed representation. To this end, it needs toderefthe object, and although special provisions are made for printing failed Agents and pending Futures, Delays are always forced.Actually I’m inclined to consider this a bug, or at best a situation in need of an improvement. As a workaround for now, take extra care not to print unforced delays.