Scala宏指定解构函数的参数

dre*_*xin 6 scala scala-macros

我目前正在玩一些宏,但也许这是一个坏主意,但这是我的问题:

我有以下宏:

def using[A <: { def close(): Unit }, B](resource: A)(f: A => B) = macro usingImpl[A, B]

def usingImpl[A <: { def close(): Unit }, B](c: Context)(resource: c.Expr[A])(f: c.Expr[A => B]): c.Expr[B] = {

  import c.universe._


  f.tree match {
    case Function(params, body) =>
      //val ValDef(modifiers, name, tpt, _) = params.head
      c.Expr[B](
        Block(
          List(
            //ValDef(modifiers, name, tpt, resource.tree)
            ValDef(params.head.symbol, resource.tree)
          ),
          body
        )
      )

    case _: Select =>
      reify {
        val res = resource.splice
        try {
          f.splice(res)
        } finally {
          res.close()
        }
      }
  }
}
Run Code Online (Sandbox Code Playgroud)

在a的情况下Select,我只需调用该函数并关闭资源,工作正常.但是在a的情况下Function,我想将param值分配给资源并调用正文.当我使用已弃用的创建者时ValDef,需要a Symbol和a Tree,一切正常.如果我使用的是out-comment 4-args创建者,我会收到编译错误,说明该值x$1不在范围内.当我查看两个版本生成的代码时,它看起来完全相同:

Expr[Int]({
  <synthetic> val x$1: Test.Foo = new Test.this.Foo();
  x$1.bar.+(23)
})
Run Code Online (Sandbox Code Playgroud)

有没有办法,简单地使用params.head和赋值?谢谢你的帮助!

编辑:

我像这样调用宏:

object Test extends App {
  import Macros._

  class Foo {
    def close() {}

    def bar = 3
  }

  println(using(new Foo)(_.bar + 3))
}
Run Code Online (Sandbox Code Playgroud)

正如我所说,如果我使用的是评论版本,它会给我一个编译器错误,它打印AST并在结尾处显示以下消息: [error] symbol value x$1 does not exist in Test$delayedInit$body.apply

我正在使用2.10.1.

Tra*_*own 7

啊,现在我可以重现你的错误.任何时候你得到"这条目似乎已经杀了编译器." 消息并在堆栈跟踪中看到这样的行:

at scala.reflect.internal.SymbolTable.abort(SymbolTable.scala:49)
Run Code Online (Sandbox Code Playgroud)

你的下一步应该是开始包装东西c.resetAllAttrs.说实话,我第一次无法重现你的错误的原因是因为我在复制和粘贴你的代码之后body在块中替换了c.resetAllAttrs(body),甚至没有想到它.这只是一种反射.

例如,参见这个小的单个抽象方法类演示,用于需要这个技巧的地方的类似实例.

在你的情况下,如果我们有这个:

import scala.language.experimental.macros
import scala.reflect.macros.Context

object Macros {
  def using[A <: { def close(): Unit }, B](resource: A)(f: A => B) =
    macro usingImpl[A, B]

  def usingImpl[A <: { def close(): Unit }, B](c: Context)
    (resource: c.Expr[A])(f: c.Expr[A => B]): c.Expr[B] = {
    import c.universe._

    val expr = f.tree match {
      case Function(ValDef(modifiers, name, tpt, _) :: Nil, body) =>
        c.Expr[B](
          Block(
            ValDef(modifiers, name, tpt, resource.tree) :: Nil,
            c.resetAllAttrs(body)
          )
        )

      case _: Select => reify {
        val res = resource.splice
        try { f.splice(res) } finally { res.close() }
      }
    }

    println(expr)
    expr
  }
}
Run Code Online (Sandbox Code Playgroud)

编译测试代码时,我们会看到以下内容:

Expr[B]({
  <synthetic> val x$1: $line3.$read.$iw.$iw.Test.Foo = new Test.this.Foo();
  x$1.bar.$plus(3)
})
Run Code Online (Sandbox Code Playgroud)

完全按照需要.