我有一个看起来像这样的对象:
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
}
答案 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
大小写以使用实例镜像,
但是从子对象递归收集键值对的基本方法是相同的。