我注意到执行以下代码时发生了AbstractMethodError。
package javapkg;
public abstract class JavaClass{
abstract String foo(int b);
public String bar(int b){
return foo(b);
}
}
package scalapkg
import javapkg._
class ScalaClass extends JavaClass{
def foo(a:Int) = a.toString
}
object Main extends App{
println(new ScalaClass().bar(10))
}
[error] (run-main) java.lang.AbstractMethodError: javapkg.JavaClass.foo(I)Ljava/lang/String;
java.lang.AbstractMethodError: javapkg.JavaClass.foo(I)Ljava/lang/String;
at javapkg.JavaClass.bar(JavaClass.java:7)
at scalapkg.Main$delayedInit$body.apply(ScalaClass.scala:10)
at scala.Function0$class.apply$mcV$sp(Function0.scala:34)
at scala.runtime.AbstractFunction0.apply$mcV$sp(AbstractFunction0.scala:12)
at scala.App$$anonfun$main$1.apply(App.scala:60)
at scala.App$$anonfun$main$1.apply(App.scala:60)
at scala.collection.LinearSeqOptimized$class.foreach(LinearSeqOptimized.scala:59)
at scala.collection.immutable.List.foreach(List.scala:76)
at scala.collection.generic.TraversableForwarder$class.foreach(TraversableForwarder.scala:30)
at scala.App$class.main(App.scala:60)
at scalapkg.Main$.main(ScalaClass.scala:9)
at scalapkg.Main.main(ScalaClass.scala)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
java.lang.RuntimeException: Nonzero exit code: 1
at scala.sys.package$.error(package.scala:27)
这是错误还是规范?
P.S。 Scala版本2.9.2,2.10.0-M3和2.10.0-SNAPSHOT(2.10.0-20120507-163905-843ac9520c)发生了同样的错误
答案 0 :(得分:6)
这有点复杂。这是一个错误 - Scala编译器应该发出错误 - 但它有点微妙。
问题在于,由于您已将public
从抽象方法中删除,因此其可见性是包私有的。如果你用Java写这个:
package javapkgimpl;
public class JavaClassImpl extends javapkg.JavaClass {
String foo(int b) { return (new java.lang.Integer(b)).toString(); }
public static void main(String[] args) {
System.out.println(new JavaClassImpl().bar(10));
}
}
然后Java编译器会抱怨:
JavaClassImpl.java:3: javapkgimpl.JavaClassImpl is not abstract and does not
override abstract method foo(int) in javapkg.JavaClass
public class JavaClassImpl extends javapkg.JavaClass {
^
1 error
显然是尝试来覆盖,但它实际上并不起作用,因为它不在正确的包中,因此无法实际访问(或覆盖)原始foo
。如果我们将包更改为javapkg
,那么一切正常。如果我们尝试使用此Scala:
package javapkg
class ScalaClass extends JavaClass{
def foo(a:Int) = a.toString
}
object ScalaClass extends App{
println(new ScalaClass().bar(10))
}
然后Scala也可以。
Scala编译器的错误是它没有注意到这个错误 - 它应该发出与Java编译器相同的错误(或者实际上告诉你访问是棘手的更有用的错误!)而不是发出字节码无法工作。
但它很容易修复:只需更改Scala代码(或Java代码中的访问修饰符)上的包。
答案 1 :(得分:0)
我发现了同样的问题。当我显式设置Scala代码返回类型时,它就起作用了。就像这样:
<div class="nr11"><h1></h1></div>