8

让我们看一下 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默默地忽略了 and 中的非关键字:req并为and:req-un抛出异常(从 alpha15 开始),我们必须以某种方式解决这个问题。:opt:opt-un

我们可以通过

(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*,它只会保存布尔值?

4

1 回答 1

8

仅供参考,目前没有计划支持s/keys.

有几种方法可以做到这一点(一种是clojure.walk/keywordize-keys在验证之前或在领先的conformer中做一些事情,然后使用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逻辑真的特殊值,但也会触发额外的行为。

于 2017-04-17T19:32:14.353 回答