如何解决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转换为some
或contains?
可以使用的东西是否有意义?如果是,如何在items
变量上执行这种转换?
更新。按照cfrick在答案中的建议使用.contains
进行了尝试,如下所示:
(defmethod call-fn "contains" [_ item items] (.contains items item))
这不会引发运行时错误,但是现在输出始终为false
。 println
调试揭示了这一点:
(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
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。