在Clojure中,如何在我自己的记录和类型上实现标准的Clojure集合接口?

Zub*_*air 18 clojure

我希望创建一个表示数据库表的抽象,但可以使用所有常用的Clojure seq和conj以及所有那些花哨的东西来访问它.我需要添加一个协议吗?

Abh*_*kar 15

是.该协议由Java接口定义clojure.lang.ISeq.您可能希望扩展clojure.lang.ASeq它提供它的抽象实现.

下面是一个示例:seq资源的抽象,可以关闭,并在结束时自动关闭seq.(未经过严格测试)

(deftype CloseableSeq [delegate-seq close-fn]
  clojure.lang.ISeq
    (next [this]
      (if-let [n (next delegate-seq)]
        (CloseableSeq. n close-fn)
        (.close this)))
    (first [this] (if-let [f (first delegate-seq)] f (.close this)))
    (more [this] (if-let [n (next this)] n '()))
    (cons [this obj] (CloseableSeq. (cons obj delegate-seq) close-fn))
    (count [this] (count delegate-seq))
    (empty [this] (CloseableSeq. '() close-fn))
    (equiv [this obj] (= delegate-seq obj))
  clojure.lang.Seqable 
    (seq [this] this)
  java.io.Closeable
    (close [this] (close-fn)))
Run Code Online (Sandbox Code Playgroud)

  • `seq`的实现似乎不对,因为当在空序列上调用`seq`时应该返回nil,而在这里你只需返回原始序列. (4认同)