2

このgist で説明されている吸血鬼のメソッドのトリックを使用する型プロバイダー マクロを作成しようとしています。これは私の実装の最小モデルです:

object DeriveFamily {
  def minimal: Any = macro DeriveFamilyMacros.minimal
}

object DeriveFamilyMacros {
  class body(tree: Any) extends StaticAnnotation

  def bodyImpl(c: Context) = {
    import c.universe._

    val field = c.macroApplication.symbol
    val bodyAnn = field.annotations.filter(_.tree.tpe <:< typeOf[body]).head
    bodyAnn.tree.children.tail.head
  }

  def minimal(c: Context): c.Tree = {
    import c.universe._
    q"""object Foobar { val _x = "X"; @DeriveFamilyMacros.body(Foobar._x) def x: String = macro DeriveFamilyMacros.bodyImpl }; Foobar"""
  }
}

これはばかげた例です。実際の実装では、このトリックを使用して、リフレクション呼び出しなしでアクセスできる派生型クラス インスタンスのバンドルを作成しようとします。class実際のバージョンとこれの両方の問題は、関数で使用すると正常に動作しますが、またはobject定義のルートで使用しようとするとコンパイラがクラッシュすることです。

object Working {
  def x = {
    val foobar = DeriveFamily.minimal
    println(foobar.x)
  }

  x // Prints "X"
}

object NotWorking {
  val foobar = DeriveFamily.minimal
  println(foobar.x)
  /*
  [trace] Stack trace suppressed: run last common/test:compileIncremental for the full output.
  [error] (common/test:compileIncremental) java.lang.IllegalArgumentException: Could not find proxy for var Foobar$module: runtime.VolatileObjectRef in List(variable Foobar$module, value foobar, Object NotWorking, package <root>) (currentOwner= value <local NotWorking> )
  */
}

これを修正するための助けは大歓迎です、ありがとう。

4

0 に答える 0