我有一个宏注释,旨在应用于类定义。它的目的是一个几乎但不完全的序列化工具。它检查类的构造函数参数,然后在伴随对象上创建一个工厂方法,该方法反过来为参数提供值。它需要知道参数的类型才能做到这一点,所以我一直在对它们调用 Context.typeCheck。
当被注解的类的构造函数接受了与自己相同类型的参数时,或者在其他类似的情况下(例如,如果类型 A 和类型 B 都被注解,并且 A 有一个参数 B,而 B 有一个参数A. 应用于形式参数的类型参数也算)。任何这些情况都会导致注解被递归调用,直到 StackOverflowError 发生。
我尝试使用“withMacrosDisabled=true”作为 c.typeCheck 的参数,虽然这解决了问题,但它引入了一个不同的问题。如果被检查的类型以前没有见过,那么编译器会记住它的定义,并且它的宏根本不会被调用。这对于自引用案例来说不是问题,但在相互引用案例中确实会发生。
所以我被困住了。有解决方法吗?我可以用 c.openMacros 解决这个问题吗?
另一个选项(如果可用)是我并不严格需要类型的完整定义,我可以只使用它的完全限定名称(scala.xml.NodeSeq 而不仅仅是 NodeSeq)。我在 AST 中获得了 TypeName,但是这些很少是完全限定的,而且我不知道如何在不进行完整的 typeCheck 的情况下获得完全限定的名称。
作为一个附带问题,“withMacrosDisabled”有什么用?如果使用它可以永久阻止在传递的树中找到的类型的所有宏扩展,而不仅仅是当前的 c.typeCheck,这似乎是一个太大的锤子。即使这实际上是你想要的,你也不能真正使用它,因为宏评估将取决于类型在它们自己的源中遇到的顺序。
编辑:考虑一下,我认为编译器应该确保每个宏都只扩展一次。在循环的情况下,如在我的示例中,所涉及的至少一个宏仍然会看到一个未完全处理的类,这在这种情况下似乎是不可避免的,因为它实际上是一个循环依赖。我想,结果类型上的标志表明宏处理不是最终的,这将是处理它的最佳方法,但这可能无法在天堂中完成。