假设我有一个函数,它接收另一个函数并返回一个函数,该函数将其获得的任何参数应用于传入的函数,并将结果放入一个向量中(这只是一个简单的示例,但希望能说明我的观点)。
我认为box函数的规范看起来像这样:
如果我给盒型函数仪器化
(defn box [f]
(fn [& args]
[(apply f args)]))
我认为box函数的规范看起来像这样:
(spec/fdef box
:args (spec/cat :function (spec/fspec :args (spec/* any?)
:ret any?))
:ret (spec/fspec :args (spec/* any?)
:ret (spec/coll-of any? :kind vector? :count 1)))
如果我给盒型函数仪器化
(spec-test/instrument)
当我使用clojure.core/+调用盒子时,会出现异常。
(box +)
ExceptionInfo Call to #'user/box did not conform to spec:
In: [0] val: ([]) fails at: [:args :function] predicate: (apply fn), Cannot cast clojure.lang.PersistentVector to java.lang.Number
:clojure.spec.alpha/args (#function[clojure.core/+])
:clojure.spec.alpha/failure :instrument
:clojure.spec.test.alpha/caller {:file "form-init4108179545917399145.clj", :line 1, :var-scope user/eval28136}
clojure.core/ex-info (core.clj:4725)
如果我正确理解了错误,那么它正在使用any?谓词并为测试生成一个PersistentVector,而clojure.core/+显然无法使用它。这意味着我可以通过将box的参数函数规范更改为以下内容来使其正常工作:
(spec/fspec :args (spec/* number?)
:ret number?)
但是如果我想同时使用clojure.core/+和clojure.string/lower-case,怎么办?
注意:为了让spec在REPL中工作,我需要
:dependencies [[org.clojure/clojure "1.9.0-alpha16"]]
:profiles {:dev {:dependencies [[org.clojure/test.check "0.9.0"]]}}
:monkeypatch-clojure-test false
在project.clj文件中和以下导入中:
(require '[clojure.spec.test.alpha :as spec-test])
(require '[clojure.spec.alpha :as spec])