最近看过Clojure Protocols的演示文稿,我对现有类型的干净方式扩展印象非常深刻。 但是,我很确定已经看到过用其他语言做类似的方法,经过一段时间后我发现它是Groovy Categories。
比较一下:
@Category(String) class StringCategory {
String lower() {
return this.toLowerCase()
}
}
use (StringCategory) {
println("TeSt".lower())
assert "test" == "TeSt".lower()
}
到Clojure Protocol等价物(取自mikera's answer below and tested in ideone.com)
(defprotocol Lowerable
(lower [x]))
(extend-protocol Lowerable
String
(lower [s]
(.toLowerCase s)))
(println (lower "HELLO"))
我的问题是:
免责声明:我是一个完整的Clojure新手!
答案 0 :(得分:10)
这是使用协议的粗略等效Clojure代码:
(defprotocol Lowerable
(lower [x]))
(extend-protocol Lowerable
String
(lower [s]
(.toLowerCase s)))
(lower "HELLO")
=> "hello"
关于Clojure协议的关键区别(我认为它与Groovy类别版本不同)
lower
来处理Rope。 lower
是一个合适的头等功能。因此,您可以使用它来构建更高阶的抽象(通过更高阶函数),这反过来将接受可扩展协议已被扩展的任何参数。 Clojure协议实际上是一个相当独特的solution to the Expression Problem(链接的视频非常有趣)。我认为在另一种语言中与Clojure协议最接近的等价物实际上是Haskell类型。即便如此,由于Haskell是静态类型的,并且Clojure是动态类型的,所以它有点延伸....
答案 1 :(得分:5)
他所指的Clojure功能如下:
(defprotocol StringMunging
(lower [this]))
(extend-protocol StringMunging
String
(lower [this]
(.toLowerCase this))
clojure.lang.Keyword
(lower [this]
(keyword (lower (name this)))))
user> (lower "TeSt")
"test"
user> (lower :TeSt)
:test
可以随时为任何类型添加实现 - 我写的两个实现都不需要以任何方式进行合作。
然而,我并不完全理解Groovy对该问题本身作出任何实质性评论;我只能帮助描述问题的Clojure方面。