2024 Clojure调查表中分享您的想法!

欢迎!请查看关于页面以了解更多关于如何工作的信息。

0
记录和类型

考虑以下代码

(definterface Test
(^void fail (link: )))

(deftype TestImpl
(link: ^{:unsynchronized-mutable true :tag int} x)
Test
(fail (link: this)

(set! x (dec x))))

它的编译失败,错误信息如下
CompilerException java.lang.VerifyError: (class: test/TestImpl, method: fail signature: ()V) Expecting to find integer on stack, compiling:(.../test.clj:27)

下面的代码能正常工作

(definterface Test
(^void fail (link: )))

(deftype TestImpl
(link: ^{:unsynchronized-mutable true :tag int} x)
Test
(fail (link: this)

(set! x (int (dec x)))))

这里唯一的改变就是我将(dec x)形式包裹在了(int)调用中。

我明白实际上前面的代码根本不可能工作(或者至少不应该按我期望的方式工作),因为(dec)被定义为调用clojure.lang.Numbers.dec(),这个方法仅对double、long和Object重载(实际上,将第一个示例中的:tag int改为:tag long可以让程序编译通过)。然而,错误信息完全无信息量且具有误导性;它看起来像是编译器错误的后果。这也不是这个具体示例的问题;我在更复杂的接口方法实现中遇到过这种错误,其中(set!)调用就在其体的正中央。

我使用Clojure 1.4.0,在Archlinux x86_64和Windows 7 x86_64上都有过这个问题。

错误的全栈跟踪信息,如果有助于解决问题

java.lang.VerifyError: (class: test/TestImpl, method: fail signature: ()V) Expecting to find integer on stack, compiling:(.../test.clj:27)

at clojure.lang.Compiler.analyzeSeq(Compiler.java:6462)
at clojure.lang.Compiler.analyze(Compiler.java:6262)
at clojure.lang.Compiler.analyze(Compiler.java:6223)
at clojure.lang.Compiler$BodyExpr$Parser.parse(Compiler.java:5618)
at clojure.lang.Compiler$FnMethod.parse(Compiler.java:5054)
at clojure.lang.Compiler$FnExpr.parse(Compiler.java:3674)
at clojure.lang.Compiler.analyzeSeq(Compiler.java:6453)
at clojure.lang.Compiler.analyze(Compiler.java:6262)
at clojure.lang.Compiler.analyzeSeq(Compiler.java:6443)
at clojure.lang.Compiler.analyze(Compiler.java:6262)
at clojure.lang.Compiler.access$100(Compiler.java:37)
at clojure.lang.Compiler$DefExpr$Parser.parse(Compiler.java:518)
at clojure.lang.Compiler.analyzeSeq(Compiler.java:6455)
at clojure.lang.Compiler.analyze(Compiler.java:6262)
at clojure.lang.Compiler.analyzeSeq(Compiler.java:6443)
at clojure.lang.Compiler.analyze(Compiler.java:6262)
at clojure.lang.Compiler.analyze(Compiler.java:6223)
at clojure.lang.Compiler$BodyExpr$Parser.parse(Compiler.java:5618)
at clojure.lang.Compiler$LetExpr$Parser.parse(Compiler.java:5919)
at clojure.lang.Compiler.analyzeSeq(Compiler.java:6455)
at clojure.lang.Compiler.analyze(Compiler.java:6262)
at clojure.lang.Compiler.analyze(Compiler.java:6223)
at clojure.lang.Compiler$BodyExpr$Parser.parse(Compiler.java:5618)
at clojure.lang.Compiler$FnMethod.parse(Compiler.java:5054)
at clojure.lang.Compiler$FnExpr.parse(Compiler.java:3674)
at clojure.lang.Compiler.analyzeSeq(Compiler.java:6453)
at clojure.lang.Compiler.analyze(Compiler.java:6262)
at clojure.lang.Compiler.eval(Compiler.java:6508)
at clojure.lang.Compiler.load(Compiler.java:6952)
at clojure.lang.Compiler.loadFile(Compiler.java:6912)
at clojure.lang.RT$3.invoke(RT.java:307)
at test$eval3224.invoke(NO_SOURCE_FILE:43)
at clojure.lang.Compiler.eval(Compiler.java:6511)
at clojure.lang.Compiler.eval(Compiler.java:6477)
at clojure.core$eval.invoke(core.clj:2797)
at clojure.main$repl$read_eval_print__6405.invoke(main.clj:245)
at clojure.main$repl$fn__6410.invoke(main.clj:266)
at clojure.main$repl.doInvoke(main.clj:266)
at clojure.lang.RestFn.invoke(RestFn.java:421)
at clojure.main$repl_opt.invoke(main.clj:332)
at clojure.main$main.doInvoke(main.clj:428)
at clojure.lang.RestFn.invoke(RestFn.java:397)
at clojure.lang.Var.invoke(Var.java:411)
at clojure.lang.AFn.applyToHelper(AFn.java:159)
at clojure.lang.Var.applyTo(Var.java:532)
at clojure.main.main(main.java:37)

原因:java.lang.VerifyError: (class: test/TestImpl, method: fail signature: ()V) Expecting to find integer on stack

at java.lang.Class.forName0(Native Method)
at java.lang.Class.forName(Class.java:264)
at clojure.lang.RT.classForName(RT.java:2039)
at clojure.lang.Compiler$HostExpr.maybeClass(Compiler.java:957)
at clojure.lang.Compiler$HostExpr.access$400(Compiler.java:736)
at clojure.lang.Compiler$NewExpr$Parser.parse(Compiler.java:2473)
at clojure.lang.Compiler.analyzeSeq(Compiler.java:6455)
... 45 more

3 个答案

0

评论由:dpx-infinity

不应该设置为高优先级;但我不能再编辑问题啦 :(:

0

评论者:jafingerhut

将优先级降低为次要,因为创建票据的用户无法自行操作。

0
参考:https://clojure.atlassian.net/browse/CLJ-1133 (由 alex+import 报告)
...