この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> )
*/
}
これを修正するための助けは大歓迎です、ありがとう。