UnmodifiableRandomAccessList和“包含”问题?功能

问题描述

我正在努力为一个名为stensil的开源应用程序做出贡献。它是用Clojure写的。

stensil是可在.docx上运行的模板引擎。模板引擎声明了将在模板中使用的一堆自定义函数功能的完整列表很短,可以here看到。

我需要添加一个自定义函数来检查列表是否包含元素。在existing examples之后,我这样做了:

(defmethod call-fn "contains" [_ item items] (contains? items item))

要对此进行测试,我创建了一个.docx文件,其中包含以下代码段:

{%= contains(5,data) %}

一个包含以下内容的.json文件

{
  "data": [9,4,1,6,3]
}

然后我像这样运行应用程序:

java -jar stencil-core-0.3.8-standalone.jar template.docx sample.json

不幸的是,使用时,我定义的功能无法正常工作,并引发以下异常:

java.lang.IllegalArgumentException: contains? not supported on type: java.util.Collections$UnmodifiableRandomAccessList

看来items的类型为java.util.Collections$UnmodifiableRandomAccessList。确实,如果我在函数添加(println items),我将得到以下信息:

#object[java.util.Collections$UnmodifiableRandomAccessList 0x778ca8ef [9,3]]

我尝试通过使用some函数来做不同的事情。 This answer建议它应该大部分时间都有效。不幸的是,下面的两个变体都返回nil

(defmethod call-fn "contains" [_ item items] (some #{item} items))
(defmethod call-fn "contains" [_ item items] (some #(= item %) items))

我前世从未写过Clojure的台词,现在有点迷失了。我究竟做错了什么?将列表从UnmodifiableRandomAccessList转换为somecontains?可以使用的东西是否有意义?如果是,如何在items变量上执行这种转换?

更新。按照cfrick在答案中的建议使用.contains进行了尝试,如下所示:

(defmethod call-fn "contains" [_ item items] (.contains items item))

这不会引发运行时错误,但是现在输出始终为falseprintln调试揭示了这一点:

(defmethod call-fn "contains" [_ item items]
  (println item)
  (println items)
  (.contains items item))
1
#object[java.util.Collections$UnmodifiableRandomAccessList 0x778ca8ef [9,3]]

为什么?

解决方法

上面的答案都是正确的,因为contains?不能用于此目的,并且我们正在尝试比较不同的类型。

此特定示例中的问题在于,第一个参数中的数字是java.lang.Long,而第二个参数中的列表包含java.math.BigDecimal实例。这是因为默认的JSON解析器模具在独立模式deserializes numbers as BigDecimal中使用。

一种快速而肮脏的解决方案是在各处使用字符串。

您还可以定义专门用于数字的功能的变体:

(defmethod call-fn "ncontains" [_ item items]
  (boolean (some #{(double item)} (map double items))))

或将所有数字强制为同一类型:

(defmethod call-fn "contains" [_ item items]
  (letfn [(norm [x] (if (number? x) (double x) x))]
    (boolean (some #{(norm item)} (map norm items)))))
,

contains?仍然不会做您想要的事情-它是用于支持查找的事情(例如,您可以检查列表是否具有 index -而不是其中是否有值) )。

从文档中

如果键在给定集合中存在,则返回true;否则返回 返回false。 请注意,对于带有数字索引的集合,例如 向量和Java数组,这将测试数字键是否在 索引范围。“包含?”操作恒定或对数时间; 它不会对值进行线性搜索。另请参阅“一些”。

您可以在这里{@ 3}退回Java所提供的内容:

如果此集合包含指定的元素,则返回true。更正式地说,当且仅当此集合包含至少一个元素(o==null ? e==null : o.equals(e))时,返回true。

因此您可以这样做:

(.contains items item)

进一步的探索:

user=> (.contains (java.util.Collections/unmodifiableList [42]) 42)
true
user=> (.contains (java.util.Collections/unmodifiableList [42]) 64)
false
user=> (.contains [42] 42)
true
user=> (.contains [42] 64)
false