Scala 2.10反射,如何从一个case class中提取字段值,即从case class中获取字段列表。

30
如何使用scala 2.10中的新反射模型来提取case class中的字段值? 例如,使用以下方法无法提取字段方法。
  def getMethods[T:TypeTag](t:T) =  typeOf[T].members.collect {
    case m:MethodSymbol => m
  }

我计划将它们泵入。
  for {field <- fields} {
    currentMirror.reflect(caseClass).reflectField(field).get
  }
2个回答

54

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
  }

啊,我现在意识到我的方法是错误的。有什么办法可以从未知的 case 类中获取 caseAccessors 吗?例如,当前存储为 val SomeCaseClass:Any。 - J Pullar
等等,不对 currentMirror.reflect(someCaseClass).symbol.asType.typeSignature.members - J Pullar
这个能在 Scala 2.10 的多线程环境下工作吗? - jilen
4
有没有办法让返回的方法按照它们被声明的顺序排序? - samthebest
13
@samthebest: 是的!只需要在members后面添加.sorted(详见 MemberScope文档)。 - Travis Brown
显示剩余3条评论

14
如果想变得更加高级,可以通过检查构造函数符号按顺序获取它们。即使所涉及的case class类型有多个定义的构造函数,此代码也可正常工作。
  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
    }
  }

我发现在表达式constructorSymbol.asTerm.alternatives中,会出现scala.ScalaReflectionException: <none> is not a term的失败情况。对于declaration的文档注释提到了OverloadedSymbol,但是似乎并不存在这样的实体。 - Randall Schulz
原来是因为我在一个作为所讨论的 case class 的超类型使用的 trait 中,使用了 this.type 调用它。 - Randall Schulz
另外,case class P(i: Int)(j: Int) - som-snytt

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接