从 Scala 调用 Java 通用类型化方法会出现类型不匹配错误:Scala

the*_*vpc 7 java generics scala scala-java-interop

这是我在这里的第一个问题,所以我会尽量说清楚。error: type mismatch;此处的其他问题与此错误无关。

我在 scala/java 互操作性方面遇到了这个奇怪的问题:

假设我们有一个 Java 类

public class JavaClass{
  public static <T> T[] toArray(Class<T> t, Collection<T> coll) {
    return null; // return null to make it simple
  }
}
Run Code Online (Sandbox Code Playgroud)

然后我有另一个 Scala 类,我只想将此方法包装在:

object ScalaClass{
  def toArray[T](t: java.lang.Class[T], coll: java.util.Collection[T]): Array[T] = {
    JavaClass.toArray[T](t, coll);
  }
}
Run Code Online (Sandbox Code Playgroud)

编译 Scala 类给了我一个非常奇怪的错误:

error: type mismatch;
Run Code Online (Sandbox Code Playgroud)

任何想法表示赞赏。

编辑 1:完整的错误消息是

[ERROR] ScalaScala.scala:4002: error: type mismatch;
[INFO]  found   : Array[T]
[INFO]  required: Class[T]
[INFO]  JavaClass.toArray[T](t, coll);
[INFO]                      ^
Run Code Online (Sandbox Code Playgroud)

Lui*_*rez 3

欢迎来到SO。

这似乎可以编译,但我不确定它是否适用于JavaClass的实际实现,但我很确定它应该。

import scala.reflect.ClassTag

object ScalaClass {
  def toArray[T <: AnyRef](coll: java.util.Collection[T])(implicit ct: ClassTag[T]): Array[T] =
    JavaClass.toArray[T](ct.runtimeClass.asInstanceOf[Class[T]], coll)
}
Run Code Online (Sandbox Code Playgroud)

要从Scala调用它,您只需要传递 java 集合,它就会从类型中获取正确的类,如下所示:

val col: java.util.Collection[String] = ???

val array: Array[String] = ScalaClass.toArray(col)
Run Code Online (Sandbox Code Playgroud)

编辑

如果你想保留原始签名,你必须这样做:

object ScalaClass {
  def toArray[T <: AnyRef](t: java.lang.Class[T], coll: java.util.Collection[T]): Array[T] =
    JavaClass.toArray[T](t, coll)
}
Run Code Online (Sandbox Code Playgroud)

你可以这样使用:

val col: java.util.Collection[String] = ???

val array: Array[String] = ScalaClass.toArray(classOf[String], col)
Run Code Online (Sandbox Code Playgroud)

在这两种情况下,技巧都是<: AnyRef 类型绑定
恕我直言,第一个版本更适合用于惯用的 Scala 代码。