scala 2.10の新しいリフレクションモデルを使用して、scalaのケースクラスからフィールド値を抽出するにはどうすればよいですか?たとえば、以下を使用してもプルされませんフィールドメソッドの外
def getMethods[T:TypeTag](t:T) = typeOf[T].members.collect {
case m:MethodSymbol => m
}
私はそれらをポンプに入れるつもりです
for {field <- fields} {
currentMirror.reflect(caseClass).reflectField(field).get
}
MethodSymbol
には、これを正確に実行できるisCaseAccessor
メソッドがあります。
def getMethods[T: TypeTag] = typeOf[T].members.collect {
case m: MethodSymbol if m.isCaseAccessor => m
}.toList
これで、次のように書くことができます。
scala> case class Person(name: String, age: Int)
defined class Person
scala> getMethods[Person]
res1: List[reflect.runtime.universe.MethodSymbol] = List(value age, value name)
そして、必要なメソッドシンボルのみを取得します。
実際のフィールド名(value
接頭辞ではない)だけが必要で、同じ順序でそれらが必要な場合:
def getMethods[T: TypeTag]: List[String] =
typeOf[T].members.sorted.collect {
case m: MethodSymbol if m.isCaseAccessor => m.name.toString
}
より洗練されたものにしたい場合は、コンストラクターシンボルを調べることで順番に取得できます。このコードは、問題のケースクラスタイプに複数のコンストラクターが定義されている場合でも機能します。
import scala.collection.immutable.ListMap
import scala.reflect.runtime.universe._
/**
* Returns a map from formal parameter names to types, containing one
* mapping for each constructor argument. The resulting map (a ListMap)
* preserves the order of the primary constructor's parameter list.
*/
def caseClassParamsOf[T: TypeTag]: ListMap[String, Type] = {
val tpe = typeOf[T]
val constructorSymbol = tpe.decl(termNames.CONSTRUCTOR)
val defaultConstructor =
if (constructorSymbol.isMethod) constructorSymbol.asMethod
else {
val ctors = constructorSymbol.asTerm.alternatives
ctors.map(_.asMethod).find(_.isPrimaryConstructor).get
}
ListMap[String, Type]() ++ defaultConstructor.paramLists.reduceLeft(_ ++ _).map {
sym => sym.name.toString -> tpe.member(sym.name).asMethod.returnType
}
}