让我们看看Leiningen项目地图的真实例子:global-vars
:
;; Sets the values of global vars within Clojure. This example
;; disables all pre- and post-conditions and emits warnings on
;; reflective calls. See the Clojure documentation for the list of
;; valid global variables to set (and their meaningful values).
:global-vars {*warn-on-reflection* true
*assert* false}
它允许leiningen的用户重新定义默认值 Clojures项目范围的全局变量。
现在,如果此地图的键由关键字组成,我们将clojure.spec/keys
首先指定哪些键可以作为地图的一部分,然后分别定义这些键下的预期值。但由于clojure.spec/keys
默默地忽略了:req
和:req-un
中的非关键字,并引发:opt
和:opt-un
的异常(从alpha15开始),我们将不得不以某种方式解决这个问题。
我们可以通过
获取大多数这些全局变量的类型(for [[sym varr] (ns-publics 'clojure.core)
:when (re-matches #"\*.+\*" (name sym))]
[varr (type @varr)])
=>
[*print-namespace-maps* java.lang.Boolean]
[*source-path* java.lang.String]
[*command-line-args* clojure.lang.ArraySeq]
[*read-eval* java.lang.Boolean]
[*verbose-defrecords* java.lang.Boolean]
[*print-level* nil]
[*suppress-read* nil]
[*print-length* nil]
[*file* java.lang.String]
[*use-context-classloader* java.lang.Boolean]
[*err* java.io.PrintWriter]
[*default-data-reader-fn* nil]
[*allow-unresolved-vars* java.lang.Boolean]
[*print-meta* java.lang.Boolean]
[*compile-files* java.lang.Boolean]
[*math-context* nil]
[*data-readers* clojure.lang.PersistentArrayMap]
[*clojure-version* clojure.lang.PersistentArrayMap]
[*unchecked-math* java.lang.Boolean]
[*out* java.io.PrintWriter]
[*warn-on-reflection* nil]
[*compile-path* java.lang.String]
[*in* clojure.lang.LineNumberingPushbackReader]
[*ns* clojure.lang.Namespace]
[*assert* java.lang.Boolean]
[*print-readably* java.lang.Boolean]
[*flush-on-newline* java.lang.Boolean]
[*agent* nil]
[*fn-loader* nil]
[*compiler-options* nil]
[*print-dup* java.lang.Boolean]
其余的我们可以通过阅读文档来填写。但我的问题是:我们如何编写一个规范,确保如果地图包含密钥'*assert*
,它只会保留布尔值?
答案 0 :(得分:4)
仅供参考,目前还没有计划支持s/keys
中的非关键字密钥。
有几种方法可以做到这一点(一种方法是在验证之前或在领先的构造函数中执行clojure.walk/keywordize-keys
之类的操作,然后使用s/keys
)。
另一条路径是将地图视为地图条目元组的集合(某些宏可以大大清除它):
(defn warn-on-reflection? [s] #(= % '*warn-on-reflection*))
(s/def ::warn-on-reflection (s/tuple warn-on-reflection? boolean?))
(defn assert? [s] #(= % '*assert*))
(s/def ::assert (s/tuple assert? boolean?))
(s/def ::global-vars
(s/coll-of (s/or :wor ::warn-on-reflection, :assert ::assert) :kind map?))
最后s/multi-spec
尝试而不是上面的s/or
可能会很有趣 - 这会使这个开放的规范可以在以后添加。以某种方式,将此规范打开可能很有用(因此也接受(s/tuple any? any?)
- 因为将来可能会添加新内容或未在此处列出(例如,有动态变量来自其他名称空间)。
另外,请注意这些属性规范。例如,*unchecked-math*
被视为逻辑上的真值,特别是采用逻辑真的特殊值:warn-on-boxed
,但也会触发额外的行为。