包私有方法覆盖时发生AbstractMethodError

Ken*_*ida 7 java overriding scala runtime-error

我注意到执行以下代码时发生了AbstractMethodError.

package javapkg;

public abstract class JavaClass{
  abstract String foo(int b);

  public String bar(int b){
    return foo(b);
  }
}
Run Code Online (Sandbox Code Playgroud)
package scalapkg

import javapkg._

class ScalaClass extends JavaClass{
  def foo(a:Int) = a.toString
}

object Main extends App{
  println(new ScalaClass().bar(10))
}
Run Code Online (Sandbox Code Playgroud)
[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)
Run Code Online (Sandbox Code Playgroud)

这是一个错误还是规范?

PS Scala版本2.9.2,2.10.0-M3和2.10.0-SNAPSHOT(2.10.0-20120507-163905-843ac9520c)发生相同的错误

Rex*_*err 7

这有点复杂.这是一个错误 - 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));
  }
}
Run Code Online (Sandbox Code Playgroud)

然后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
Run Code Online (Sandbox Code Playgroud)

它显然试图覆盖,但它实际上并不起作用,因为它不在正确的包中,因此无法实际访问(或覆盖)原始foo.如果我们将包更改为javapkg,那么一切正常.如果我们尝试这个Scala:

package javapkg

class ScalaClass extends JavaClass{
  def foo(a:Int) = a.toString
}

object ScalaClass extends App{
  println(new ScalaClass().bar(10))
}
Run Code Online (Sandbox Code Playgroud)

然后Scala也可以正常工作.

这是Scala编译器的错误,它没有注意到这个错误 - 它应该发出与Java编译器相同的错误(或者更实用的一个告诉你访问是棘手的!)而不是发出不能的字节码工作.

但它很容易修复:只需更改Scala代码(或Java代码中的访问修饰符)上的包.