Подтвердить что ты не робот

Scala 2.10 отражение, как извлечь значения поля из класса case

Как извлечь значения полей из класса case в scala, используя новую модель отражения в scala 2.10? Например, используя ниже, не вытаскивают методы поля

  def getMethods[T:TypeTag](t:T) =  typeOf[T].members.collect {
    case m:MethodSymbol => m
  }

Я планирую перекачивать их в

  for {field <- fields} {
    currentMirror.reflect(caseClass).reflectField(field).get
  }
4b9b3361

Ответ 1

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)

И вы получите только те символы метода, которые вы хотите.

Ответ 2

Если вы хотите получить fancier, вы можете получить их в порядке, проверив символ конструктора. Этот код работает, даже если рассматриваемый тип класса case имеет несколько конструкторов.

  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 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
    }
  }