通过 Java 反射找到 Scala 类型的伴生对象的可靠方法是啥?

Posted

技术标签:

【中文标题】通过 Java 反射找到 Scala 类型的伴生对象的可靠方法是啥?【英文标题】:What is a reliable way to find a Scala type's companion object via Java reflection?通过 Java 反射找到 Scala 类型的伴生对象的可靠方法是什么? 【发布时间】:2016-03-17 17:48:38 【问题描述】:

我正在编写 Java(不是 Scala)代码,其中有一个 Java Class 引用,指的是 Scala 类型 T

trait T 

我现在想通过仅使用 Java 反射来发现 Java Class 引用以及与 object T 对应的单例实例,然后调用该伴随对象中的方法。

object T 
  def someMethod : String = "abc";

我对生成的 Scala 代码进行了逆向工程,以编写以下 Java 代码:

Class<?> traitClass = ...;
Class<?> companionClass = Class.forName(traitClass.getName() + "$");
Field module = companionClass.getField("MODULE$");
Object companion = module.get(companionClass);
String abc = (String) companionClass.getMethod("someMethod").invoke(companion);

这看起来不是很健壮——它可能会随着未来的 Scala 编译器而改变。有没有更健壮的方法?我也愿意调用 Scala API(来自 Java)来发现配套实例。

【问题讨论】:

嗯。那么您是否尝试过使用 scala.meta ?这里:github.com/scalameta/scalameta. @SomBhattacharyya:我不想为此使用库。特别不是 Scala 库,因为我在 Java 中实现它 哦,好的,我明白了。老实说 Scala 改变他们的二进制版本的方式,我认为很难写出“更好”或“可持续”的东西。 @SomBhattacharyya:我害怕那个。但也很好奇其他人是否遇到过这个问题,并以某种方式解决了它 是的,我也很想知道。很好的问题。 【参考方案1】:

您编写的代码适用于***(包拥有的)同伴。它也适用于嵌套类/特征对象对的基本情况,但有些情况下简单的name + $ 方案不成立。这是和示例:

class C 
  class D
  object D
  def foo() 
    class K
    object K
  

这会生成类文件CC$DC$D$C$K$2C$K$3$

虽然编码可能在未来的任何时间发生变化,但它在很长一段时间内都没有变化,也没有计划改变。我刚试了2.5.0,除了嵌套在方法中的类,类名都是一样的。

如果您可以控制 Scala 类,最简单的方法是将方法 def companion = T 添加到 trait T

另一种选择是使用 Scala 反射 API:

scala> import scala.reflect.runtime.universe => ru
scala> val m = ru.runtimeMirror(getClass.getClassLoader)

scala> val d = classOf[C$D]
d: Class[C$D] = class C$D

scala> m.runtimeClass(m.classSymbol(d).companionSymbol.asModule.moduleClass.asClass)
res16: Class[_] = class C$D$

但这在嵌套在方法中的类上也不能按预期工作:

scala> val k = classOf[C$K$2]
k: Class[C$K$2] = class C$K$2

scala> m.classSymbol(k).companionSymbol
res2: reflect.runtime.universe.Symbol = <none>

请注意,在 SO 上搜索“scala 反射伴侣”也会产生几个结果(并显示一些问题)。

【讨论】:

优秀。如果这在很长一段时间内都没有改变,我怀疑我的方法对于我的用例来说是“足够好”。关于 “如果您可以控制 Scala 类,最简单的方法是添加方法 def partner = T 到 trait T。” - 不幸的是,我没有 instance i> 的特征 T,只有它的 Class 引用。所以,我不能真正调用任何这样的方法(除非我可以强制 Scala 在 trait 上生成静态方法)-“请注意,在 SO 上搜索“scala 反射伴侣”也会产生几个结果” i> 是的,但通常是从 Scala 中执行此操作,而不是从 Java :)

以上是关于通过 Java 反射找到 Scala 类型的伴生对象的可靠方法是啥?的主要内容,如果未能解决你的问题,请参考以下文章

Scala编程之伴生对象

聊聊 Scala 的伴生对象及其意义

scala伴生类和伴生对象的理解

快学Scala 第九课 (伴生对象和枚举)

scala成长之路compaion object——伴生对象的使用

每天学一点Scala之 伴生类和伴生对象