我需要一个字符串名称定义的对象(或“单例对象”或“伴随对象”......除了类之外的任何东西)。换句话说,如果我有:
package myPackage
object myObject
......那就是这样的:
GetSingletonObjectByName("myPackage.myObject") match {
case instance: myPackage.myObject => "instance is what I wanted"
}
答案 0 :(得分:43)
在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)
答案 1 :(得分:12)
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)
要获取无类型的伴随对象,您可以直接使用该类:
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)
这取决于scala映射到java类的方式。
答案 2 :(得分: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)
答案 3 :(得分:1)
禁止反思技巧,你不能。请注意,例如,如何在Scala 2.8集合上定义方法companion
- 它就在那里,所以类的实例可以获取伴随对象,否则这是不可能的。