访问案例 class 注释

Accessing a case class annotation

case class FieldName(field: String) extends scala.annotation.StaticAnnotation
@FieldName("foo") trait Foo
import scala.reflect.runtime.universe._

symbolOf[Foo].annotations.head
// ann: Annotation = FieldName("type")

如何将注释作为 FieldName 对象访问?文档提到 tree.children.tail,但没有类型。

如果你真的想要 FieldName 的实例,我能想到的最好的方法是使用 ToolBox:

scala> case class FieldName(field: String) extends scala.annotation.StaticAnnotation
defined class FieldName

scala> @FieldName("foo") trait Foo
defined trait Foo

scala> import scala.reflect.runtime.universe._
import scala.reflect.runtime.universe._

scala> val annotation = symbolOf[Foo].annotations.head
annotation: reflect.runtime.universe.Annotation = FieldName("foo")

scala> import scala.tools.reflect.ToolBox
import scala.tools.reflect.ToolBox

scala> val tb = runtimeMirror(getClass.getClassLoader).mkToolBox()
tb: scala.tools.reflect.ToolBox[reflect.runtime.universe.type] = scala.tools.reflect.ToolBoxFactory$ToolBoxImpl@1b26a499

scala> tb.eval(tb.untypecheck(annotation.tree)).asInstanceOf[FieldName]
res10: FieldName = FieldName(foo)

使用 .tree.children.tail,您可以访问传递给 FieldName 的参数,而无需创建实际实例。

scala> annotation.tree.children.tail.map{ case Literal(Constant(field)) => field } 
res11: List[Any] = List(foo)

如果您只想要所有 FieldName 注释并提取它们的值,您可以这样做:

scala> val fields = symbolOf[Foo].annotations.withFilter( 
     |   a => a.tree.tpe <:< typeOf[FieldName] 
     | ).flatMap( 
     |   a => a.tree.children.tail.map{ case Literal(Constant(field)) => field } 
     | )
fields: List[Any] = List(foo)