Schema 中有两个独立的概念:验证和强制。
对于第一个,您需要定义您的架构。类被视为模式,因此您无需为java.math.BigDecimal 创建自定义模式。您的架构可能如下所示:
(require '[schema.core :as s])
(require '[schema.coerce :as c])
(s/defschema Loan {:loan/amount java.math.BigDecimal})
现在您可以根据架构验证您的数据:
(s/validate Loan {:loan/amount 10M})
;; => {:loan/amount 10M}
现在,如果您有一些想要强制的数据,则需要定义一个强制函数,该函数是从所需目标模式(在您的情况下为 java.math.BigDecimal)到将实际值转换为所需的bigdec 值。
(def safe-bigdec (c/safe bigdec)
schema.coerce/safe 是一个实用函数,它包装了原始函数,如果原始函数在调用时抛出异常,safe 将返回原始输入值而不是抛出异常。
我们的 matcher 函数将检查当前 schema 元素是否为 BigDecimal 并返回转换函数或 nil 否则(意味着对其他类型没有强制):
(defn big-decimal-matcher [schema]
(when (= java.math.BigDecimal schema)
safe-bigdec))
最后我们需要一个强制器来执行实际的强制:
(def loan-coercer (c/coercer Loan big-decimal-matcher))
完成所有设置后,我们现在可以使用我们的强制器了:
(loan-coercer {:loan/amount "12.34"})
;; => {:loan/amount 12.34M}
(loan-coercer {:loan/amount 1234})
;; => {:loan/amount 1234M}
(loan-coercer {:loan/amount "abc"})
;; => #schema.utils.ErrorContainer{:error {:loan/amount (not (instance? java.math.BigDecimal "abc"))}}