8

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

这是错误还是规范?

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

4

2 回答 2

8

这有点复杂。这是一个错误——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 代码中的访问修饰符)。

于 2012-05-11T18:19:59.457 回答
0

我发现了同样的问题。当我明确设置 Scala 代码返回类型时,它起作用了。就像:

class ScalaClass extends JavaClass{
  def foo(a:Int):String = a.toString
}
于 2019-09-03T09:45:42.283 回答