如何在Clojure中使用 *print-dup* 打印记录?一个简单的案例

5
我尝试使用 *print-dup* 将 Clojure 数据写入文件并读取,但即使在这个简单的案例中,我也遇到了问题。我做错了什么吗?我需要怎么做才能让它工作?
Clojure 1.3.0-alpha3-SNAPSHOT 
user=> (defrecord TreeNode [val left right]) ;;create the record 
user.TreeNode 
user=> (TreeNode. 5 nil nil) 
#:user.TreeNode{:val 5, :left nil, :right nil} ;; it works just fine 
user=> (binding [*print-dup* true] (prn (TreeNode. 5 nil nil))) ;; use *print-dup* to support reading in and preserving type 
#=(user.TreeNode/create {:val #=(java.lang.Long. "5"), :left nil, :right nil}) ;; this is the form we need to copy paste 
nil 
user=> #=(user.TreeNode/create {:val #=(java.lang.Long. "5"), :left nil, :right nil}) ;;trying to copy and paste 
IllegalArgumentException No matching method found: create 
clojure.lang.Reflector.invokeMatchingMethod (Reflector.java:50) ;;we have an error 
user=>
3个回答

6
记录不以可评估的形式打印。我们添加了自己的defrecord2,包括创建构造函数和print/pprint支持的支持,以使用构造函数打印它们(可以将其eval回原始记录)。这在此处此处有更多文档,并且代码在这里。
我在2010年10月的Clojure Conj会议上向Rich Hickey提出了这个问题,他说构造函数和记录的reader支持计划在未来实现。

1
设计未来版本中记录的工作在这里进行:http://dev.clojure.org/display/design/defrecord+improvements - Alex Miller

4
作为更新,从alpha8开始,在简单情况下,使用*print-dup*与记录一起工作。
 user=> (defrecord TreeNode [val left right])
 user.TreeNode
 user=> (TreeNode. 5 nil nil)
 #user.TreeNode{:val 5, :left nil, :right nil}
 user=> (binding [*print-dup* true] (prn (TreeNode. 5 nil nil)))
 #user.TreeNode[5, nil, nil]
 nil
 user=> #user.TreeNode[5, nil, nil]
 #user.TreeNode{:val 5, :left nil, :right nil}
 user=> 

2

1.2版本的解决方法:

(defn- extend-print-dup [record-class]
  (defmethod print-dup record-class [o w]
    (.write w "#=(")
    (.write w (.getName ^Class (class o)))
    (.write w ". ")
    (dorun (map (fn [a]  (print-dup a w) (.write w " ")) (vals o)))
    (.write w ")")))

(defrecord Hl7Field [protospec segname fname pos hl7type values])
(extend-print-dup Hl7Field)

=> (def a (Hl7Field. "p" "PV1" "toto" 0 "ST"  ["c" "d"]))
=> (binding [*print-dup* true] (prn a))
#=(higiebus.bus.protocol.hl7.Hl7Field. "p" "PV1" "toto" 0 "ST" ["c" "d"] )

=> #=(higiebus.bus.protocol.hl7.Hl7Field. "p" "PV1" "toto" 0 "ST" ["c" "d"] )
{:protospec "p", :segname "PV1", :fname "toto", :pos 0, :hl7type "ST", :values ["c" "d"]}

你可以使用自定义的defrecord宏来包装对extend-print-dup的调用。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接