包私有方法覆盖时发生AbstractMethodError

时间:2012-05-11 17:14:35

标签: java scala runtime-error method-overriding

我注意到执行以下代码时发生了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)发生了同样的错误

2 个答案:

答案 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>