gpt4 book ai didi

scala - 编译器将结构类型用于在白盒宏内生成的案例类

转载 作者:行者123 更新时间:2023-12-04 18:37:47 25 4
gpt4 key购买 nike

以下宏生成一个案例类 Person 并返回此类的一个实例。它使用白盒宏,因此编译器可以推断类型 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$1: 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 .我在这里发布代码以供将来引用。

在客户端代码中,我注释了一个包装器对象,然后宏将扩展该对象以包含一个新的案例类 Person 和该类的一个实例。与使用结构类型和反射的匿名类型提供程序的解决方案相反,生成的代码使用标准方法调用。

@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)
}
}
}

关于scala - 编译器将结构类型用于在白盒宏内生成的案例类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30046230/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com