我正在尝试理解以下博客文章,其中讨论了如何使用宏来创建基于宏的通用方法,以将案例类对象转换为地图和从地图转换:http://blog.echo.sh/post/65955606729/exploring-scala-macros-map-to-case-class-conversion
即使我阅读了整篇文章和另一篇关于scala宏的文章,我仍然不完全了解它们是如何工作的。不幸的是,我从博客文章中给出的示例代码中收到了一些编译错误,我不确定错误发生的原因以及如何解决错误。为了完整性,我正在复制下面的代码,前面是编译错误(我在代码片段中注释了有关编译错误的注释)。有谁知道为什么这些编译器问题正在发生以及如何解决它们?
-type mismatch; found : field.NameType required:
c.universe.TermName
-Can't unquote List[Nothing] with .., bottom type values often indicate programmer mistake
以下是代码:
import scala.reflect.macros.Context
import scala.language.experimental.macros
trait Mappable[T] {
def toMap(t: T): Map[String, Any]
def fromMap(map: Map[String, Any]): T
}
object Mappable {
implicit def materializeMappable[T]: Mappable[T] = macro materializeMappableImpl[T]
def materializeMappableImpl[T: c.WeakTypeTag](c: Context): c.Expr[Mappable[T]] = {
import c.universe._
val tpe = weakTypeOf[T]
val companion = tpe.typeSymbol.companionSymbol
val fields = tpe.declarations.collectFirst {
case m: MethodSymbol if m.isPrimaryConstructor ⇒ m
}.get.paramss.head
val (toMapParams, fromMapParams) = fields.map { field ⇒
val name = field.name
val decoded = name.decoded
val returnType = tpe.declaration(name).typeSignature
(q"$decoded → t.$name", q"map($decoded).asInstanceOf[$returnType]") // error 1
}.unzip
c.Expr[Mappable[T]] {
q"""
new Mappable[$tpe] {
def toMap(t: $tpe): Map[String, Any] = Map(..$toMapParams)
def fromMap(map: Map[String, Any]): $tpe = $companion(..$fromMapParams) // error 2
}
"""
}
}
}
答案 0 :(得分:1)
将t。$ name更改为t。$ {name.toTermName}