从案例类中获取字段名称列表

Ale*_*rov 36 scala

我只需要获取案例类的字段名称.我对它的价值观不感兴趣.我以为getClass.getDeclaredFields.map(_.getName)会返回一个字段名称列表.

scala> case class User(id: Int, name: String)
defined class User

scala> User.getClass.getDeclaredFields
res14: Array[java.lang.reflect.Field] = Array(public static final User$ User$.MODULE$)

scala> User.getClass.getDeclaredFields.toList
res15: List[java.lang.reflect.Field] = List(public static final User$ User$.MODULE$)

scala> val user = User(1, "dude")
user: User = User(1,dude)

scala> user.getClass.getDeclaredFields.toList
res16: List[java.lang.reflect.Field] = List(private final int User.id, private final java.lang.String User.name)
Run Code Online (Sandbox Code Playgroud)

什么是这个用户$ .MODULE $?那是什么?

当你有一个case类的实例时,方法getDeclaredFields可以正常工作,但是我不想创建一个实例以便只获取字段.

为什么这不是真的: User.getClass.getDeclaredFields.map(_.getName) == List("id", "name")

Dia*_*rat 58

通过使用User.getClass,您指的是默认情况下Scala为案例类创建的类伴随对象,而不是案例类本身.要获取案例类的类对象,请使用classOf[User].

或者,您可以使用Scala的反射API来获取案例类的元数据,从而为您提供更多信息:

import scala.reflect.runtime.universe._

def classAccessors[T: TypeTag]: List[MethodSymbol] = typeOf[T].members.collect {
  case m: MethodSymbol if m.isCaseAccessor => m
}.toList
Run Code Online (Sandbox Code Playgroud)

在sbt控制台中测试:

scala> case class User(name: String, age: Int)
defined class User

scala> classAccessors[User]
res0: List[reflect.runtime.universe.MethodSymbol] = List(value age, value name)
Run Code Online (Sandbox Code Playgroud)

  • 这个答案有一个线程解释了如何排序:http://stackoverflow.com/a/16079804/1586965 (5认同)
  • 他们似乎以lexo的顺序回来了.有没有办法让它们按照它们在代码中声明的顺序返回? (2认同)

Xav*_*hot 17

开始Scala 2.13,case classes(这是一个实现Product)现在提供了productElementNames方法,它返回了自己的字段名称的迭代器.

从案例类的一个实例(比如说case class Person(name: String, age: Int)),可以检索List它的一个字段:

Person("hello", 28).productElementNames.toList
// List[String] = List(name, age)
Run Code Online (Sandbox Code Playgroud)

  • 那很棒!哦,我在火花.. 2.13在火花4,这将是3年 (5认同)
  • 该方法令人沮丧的事情是您需要相关类型的实例,这在某些情况下不太理想 (4认同)

dre*_*xin 11

User.getClass没有给你相同的User.classJava,但它给你类的伴随对象的User类.您可以检索with 的Class对象.UserclassOf[User]

编辑:哦,User$.MODULE$它是内部使用的单例实例的访问器.可以把它想象成MyClass.INSTANCE当你用Java编写单例时的等价物.


And*_*kin 6

如果您还想知道为什么某些Scala反射代码不再编译,这是具有良好Java反射效果的粗略解决方案(显然,自大约1300年以来,它一直以完全相同的方式工作):

case class User(b: Int, a: String)
val u = User(42, "JohnDoe")

classOf[User]
.getDeclaredFields
.map{ f => 
  f.setAccessible(true)
  val res = (f.getName, f.get(u))
  f.setAccessible(false)
  res
}
Run Code Online (Sandbox Code Playgroud)

它将获取名称和值:

Array((b,42), (a,bob))
Run Code Online (Sandbox Code Playgroud)

顺序似乎与构造函数中的顺序相同。


Jav*_*tón 5

遵循 Andrey Tyukin 解决方案,仅获取 Scala 2.12 中的字段列表:

val fields: List[String] = classOf[Dummy].getDeclaredFields.map(_.getName).toList
Run Code Online (Sandbox Code Playgroud)