编译器将结构类型用于在白盒宏内生成的 case class
Compiler uses structural types for case class generated inside a whitebox macro
下面的宏生成一个案例 class Person 和 returns 这个 class 的一个实例。它使用白盒宏,因此编译器可以推断类型 Person。这允许宏客户端调用 p.name,即使该字段是在宏内部生成的。
import scala.language.experimental.macros
class MacroDefs(val c: scala.reflect.macros.whitebox.Context) {
import c.universe._
def createCaseImpl(): c.Expr[Product] = {
val code: Tree = q""" case class Person(name:String); new Person("Joe") """
c.Expr(code)
}
}
object Macros {
def createCase(): Product = macro MacrosDef.createCaseImpl
}
object Test {
def main(args: Array[String]) {
val p = Macros.createCase()
println("Name: " + p.name)
}
}
代码有效,但编译器正在使用结构类型访问 p.name,正如您从下面的警告消息中看到的(我通过反编译生成的字节码确认了这一点):
Warning:(5, 54) inferred existential type Person forSome { type Person <: Product with Serializable{val name: String; def copy(name: String): Person; def copy$default: String @scala.annotation.unchecked.uncheckedVariance} }, which cannot be expressed by wildcards, should be enabled
by making the implicit value scala.language.existentials visible.
val c = Macros.createCase()
^
由于结构类型依赖于 Java 反射,我很关心性能。
我的问题是是否有更好的方法可以让编译器使用标准方法调用而不是结构类型和反射。我正在使用 Scala 2.11.6.
第二个小问题:Intellij 似乎不能很好地处理白盒宏并将对 p.name 的访问标记为无效,表示该字段未知,即使它会编译并且 运行 与 scalac。有没有办法让 Intellij 知道白盒宏?
根据@MartinRing 和@TravisBrown 的建议,我使用宏注释解决了这个问题。这篇文章也很有用:http://www.47deg.com/blog/scala-macros-annotate-your-case-classes。我在这里发布代码以供将来参考。
在客户端代码中,我注释了一个包装器对象,然后宏将扩展该对象以包含一个新案例 class Person 和此 class 的一个实例。与使用结构类型和反射的匿名类型提供程序的解决方案相反,生成的代码使用标准方法调用。
@personGenerator object Wrapper
object Main {
def main(args: Array[String]) {
println("Name: " + Wrapper.thePerson.name)
val other = new Wrapper.Person("Joan")
println("Other: " + other.name)
}
}
下面是注释和宏的实现:
class personGenerator extends StaticAnnotation {
def macroTransform(annottees: Any*): Any = macro MacroDefs.personGenerator_impl
}
class MacroDefs(val c: scala.reflect.macros.whitebox.Context) {
import c.universe._
def personGenerator_impl(annottees: c.Expr[Any]*): c.Expr[Any] = {
annottees.map(_.tree) match {
case List(q"object $name { ..$body }") =>
val code = q"""
object $name {
..$body
case class Person(name:String)
def thePerson = new Person("Joe")
}
"""
c.Expr[Any](code)
}
}
}
下面的宏生成一个案例 class Person 和 returns 这个 class 的一个实例。它使用白盒宏,因此编译器可以推断类型 Person。这允许宏客户端调用 p.name,即使该字段是在宏内部生成的。
import scala.language.experimental.macros
class MacroDefs(val c: scala.reflect.macros.whitebox.Context) {
import c.universe._
def createCaseImpl(): c.Expr[Product] = {
val code: Tree = q""" case class Person(name:String); new Person("Joe") """
c.Expr(code)
}
}
object Macros {
def createCase(): Product = macro MacrosDef.createCaseImpl
}
object Test {
def main(args: Array[String]) {
val p = Macros.createCase()
println("Name: " + p.name)
}
}
代码有效,但编译器正在使用结构类型访问 p.name,正如您从下面的警告消息中看到的(我通过反编译生成的字节码确认了这一点):
Warning:(5, 54) inferred existential type Person forSome { type Person <: Product with Serializable{val name: String; def copy(name: String): Person; def copy$default: String @scala.annotation.unchecked.uncheckedVariance} }, which cannot be expressed by wildcards, should be enabled
by making the implicit value scala.language.existentials visible.
val c = Macros.createCase()
^
由于结构类型依赖于 Java 反射,我很关心性能。
我的问题是是否有更好的方法可以让编译器使用标准方法调用而不是结构类型和反射。我正在使用 Scala 2.11.6.
第二个小问题:Intellij 似乎不能很好地处理白盒宏并将对 p.name 的访问标记为无效,表示该字段未知,即使它会编译并且 运行 与 scalac。有没有办法让 Intellij 知道白盒宏?
根据@MartinRing 和@TravisBrown 的建议,我使用宏注释解决了这个问题。这篇文章也很有用:http://www.47deg.com/blog/scala-macros-annotate-your-case-classes。我在这里发布代码以供将来参考。
在客户端代码中,我注释了一个包装器对象,然后宏将扩展该对象以包含一个新案例 class Person 和此 class 的一个实例。与使用结构类型和反射的匿名类型提供程序的解决方案相反,生成的代码使用标准方法调用。
@personGenerator object Wrapper
object Main {
def main(args: Array[String]) {
println("Name: " + Wrapper.thePerson.name)
val other = new Wrapper.Person("Joan")
println("Other: " + other.name)
}
}
下面是注释和宏的实现:
class personGenerator extends StaticAnnotation {
def macroTransform(annottees: Any*): Any = macro MacroDefs.personGenerator_impl
}
class MacroDefs(val c: scala.reflect.macros.whitebox.Context) {
import c.universe._
def personGenerator_impl(annottees: c.Expr[Any]*): c.Expr[Any] = {
annottees.map(_.tree) match {
case List(q"object $name { ..$body }") =>
val code = q"""
object $name {
..$body
case class Person(name:String)
def thePerson = new Person("Joe")
}
"""
c.Expr[Any](code)
}
}
}