如何在Scala中使用反射获取所有对象值和子对象值?

时间:2013-10-01 14:02:45

标签: scala object reflection hash

我有一个看起来像这样的对象:

object Settings {
  final val Host = "host"
  final val Protocol = "protocol"

  object User {
    final val Name = "username"
    final val Password = "password"
  }

  object Subject {
    final val Query = "query"
    final val Predicate = "predicate"
  }
}

我想要做的是membersAsHash(classOf[CollectionSettings])并且接收哈希)我在对象中声明的所有val:

[
  Host => "host", 
  Protocol => "protocol", 
  Name => "username",
  Password => "password",
  Query => "query",
  Predicate => "predicate"
]

如果密钥是字符串,即使是完整的包名称(例如com.example.Settings.User)也没关系。我真正需要的是价值观,所以如果我只能得到它,它仍然可以接受。

这让我得到了子对象的名字,但我似乎无法弄清楚如何获得每个对象内部的val:

val optionsToCheck = {
  import scala.reflect.runtime.{universe => ru}
  val mirror = ru.runtimeMirror(getClass.getClassLoader)
  val subObjects = ru.typeOf[CollectionSettings.type].declarations.filter(_.isModule)
  subobjects.map(o => mirror.reflectModule(o.asModule).instance.asInstanceOf[Object].toString).toList
}

1 个答案:

答案 0 :(得分:9)

这里的好处是你正在使用常量值定义(即没有类型注释的最终值;参见language specification的§4.1),所以你没有甚至需要任何镜子:

def deepMembers[A: scala.reflect.runtime.universe.TypeTag](a: A) = {
  import scala.reflect.runtime.universe._

  def members(s: Symbol): Map[String, String] =
    s.typeSignature.declarations.collect {
      case m: ModuleSymbol => members(m)
      case m: MethodSymbol if m.isAccessor => m.returnType match {
        case ConstantType(Constant(s: String)) => Map(m.name.decoded -> s)
        case _ => Map.empty[String, String]
      }
    }.foldLeft(Map.empty[String, String])(_ ++ _)

  members(typeOf[A].termSymbol)
}

它的工作原理如下:

scala> deepMembers(Settings) foreach println
(Name,username)
(Predicate,predicate)
(Query,query)
(Password,password)
(Protocol,protocol)
(Host,host)

如果由于某种原因您无法使用常量值定义,则需要调整MethodSymbol大小写以使用实例镜像, 但是从子对象递归收集键值对的基本方法是相同的。