在scala中通过字符串名称获取对象实例

Dav*_*ave 27 reflection singleton scala

我需要由字符串名称定义的对象(或"单例对象"或"伴随对象"......除了类之外的任何东西).换句话说,如果我有:

package myPackage
object myObject
Run Code Online (Sandbox Code Playgroud)

...那么有这样的事情:

GetSingletonObjectByName("myPackage.myObject") match {
  case instance: myPackage.myObject => "instance is what I wanted"
}
Run Code Online (Sandbox Code Playgroud)

小智 47

在scala 2.10中,我们可以这样做

import scala.reflect.runtime.universe

val runtimeMirror = universe.runtimeMirror(getClass.getClassLoader)

val module = runtimeMirror.staticModule("package.ObjectName")

val obj = runtimeMirror.reflectModule(module)

println(obj.instance)
Run Code Online (Sandbox Code Playgroud)


Tho*_*ung 13

Scala仍缺少反射API.您可以通过加载伴随对象类来获取伴随对象的实例:

import scala.reflect._
def companion[T](implicit man: Manifest[T]) : T = 
  man.erasure.getField("MODULE$").get(man.erasure).asInstanceOf[T]


scala> companion[List$].make(3, "s")
res0: List[Any] = List(s, s, s)
Run Code Online (Sandbox Code Playgroud)

要获取无类型的伴随对象,您可以直接使用该类:

import scala.reflect.Manifest
def companionObj[T](implicit man: Manifest[T]) = { 
  val c = Class.forName(man.erasure.getName + "$")
  c.getField("MODULE$").get(c)
}


scala> companionObj[List[Int]].asInstanceOf[List$].make(3, "s")
res0: List[Any] = List(s, s, s)
Run Code Online (Sandbox Code Playgroud)

这取决于scala映射到java类的方式.

  • 天啊.你知道这个语法是否是Scala规范的固定部分(与语言中的任何其他内容一样固定,无论如何)?依靠这个似乎是一个坏主意.因为我的目标是使代码*更清晰*...谢谢! (2认同)

psp*_*psp 11

调整Thomas Jung上面的答案:你最好说同伴[List.type],因为a)这应该是一个稳定的方式来引用它,而不是依赖于名称修改方案和b)你得到了未被删除的类型.

def singleton[T](implicit man: reflect.Manifest[T]) = {
  val name = man.erasure.getName()
  assert(name endsWith "$", "Not an object: " + name)
  val clazz = java.lang.Class.forName(name)

  clazz.getField("MODULE$").get(clazz).asInstanceOf[T]
}  

scala> singleton[List.type].make(3, "a")                       
res0: List[java.lang.String] = List(a, a, a)
Run Code Online (Sandbox Code Playgroud)