Scala метод компиляции представления locateTree
Я использовал API компилятора представления Scala, или, если быть более точным, это locateTree
метод, чтобы получить AST некоторого фрагмента исходного кода, а затем получить его необработанное представление через showRaw(ast)
позвоните, но результат, кажется, отличается от того, что я ожидал. Например
val tree = q"final def x = 1"
println(showRaw(tree))
Выходы DefDef(Modifiers(FINAL), TermName("x"), List(), List(), TypeTree(), Literal(Constant(1)))
в то время как вызов компилятора представления из того же источника производит DefDef(Modifiers(32, , List()), x, List(), List(), TypeTree(), Literal(Constant(1)))
(обратите внимание, что х не завернут в TermName
и разница в Modifiers
список параметров). Почему это происходит и как я могу применить подобное поведение в компиляторе презентаций?
Изменить: версия Scala 2.11.8
1 ответ
Деревья набраны по-разному, поэтому вы должны использовать правильный show
:
$ scala
Welcome to Scala 2.11.8 (Java HotSpot(TM) 64-Bit Server VM, Java 1.8.0_60).
Type in expressions for evaluation. Or try :help.
scala> import scala.tools.nsc._
import scala.tools.nsc._
scala> val ss = new Settings(println)
ss: scala.tools.nsc.Settings =
Settings {
-d = .
}
scala> val g = new interactive.Global(ss, new reporters.ConsoleReporter(ss), "testing")
g: scala.tools.nsc.interactive.Global = scala.tools.nsc.interactive.Global@4c6007fb
scala> val tt = g.newUnitParser("final def x = 1").parseStats
tt: List[g.syntaxAnalyzer.global.Tree] = List(final def x = 1)
scala> reflect.runtime.universe.show(tt.head)
res0: String = final def x = 1
scala> reflect.runtime.universe.showRaw(tt.head)
res1: String = DefDef(Modifiers(32, , List()), x, List(), List(), TypeTree(), Literal(Constant(1)))
scala> g.showRaw(tt)
res2: String = List(DefDef(Modifiers(FINAL), TermName("x"), List(), List(), TypeTree(), Literal(Constant(1))))
Там нет тип безопасности на res1
так что вы не понимаете, что дерево из другой вселенной.
Деревья немного отличаются в модификаторах из-за аннотаций положения:
$ scala
Welcome to Scala 2.11.8 (Java HotSpot(TM) 64-Bit Server VM, Java 1.8.0_60).
Type in expressions for evaluation. Or try :help.
scala> import scala.tools.nsc._
import scala.tools.nsc._
scala> val ss = new Settings(println)
ss: scala.tools.nsc.Settings =
Settings {
-d = .
}
scala> new interactive.Global(ss, new reporters.ConsoleReporter(ss), "testing")
res0: scala.tools.nsc.interactive.Global = scala.tools.nsc.interactive.Global@4c6007fb
scala> val cc = res0
cc: scala.tools.nsc.interactive.Global = scala.tools.nsc.interactive.Global@4c6007fb
scala> val tt = cc.newUnitParser("final def x = 1").parseStats
tt: List[cc.syntaxAnalyzer.global.Tree] = List(final def x = 1)
scala> val cc.DefDef(mods, nam, ps, vs, tpt, rhs) = tt.head
mods: cc.Modifiers = Modifiers(final, , Map(32 -> RangePosition(<console>, 0, 0, 4), 72 -> RangePosition(<console>, 6, 6, 8)))
nam: cc.TermName = x
ps: List[cc.TypeDef] = List()
vs: List[List[cc.ValDef]] = List()
tpt: cc.Tree = <type ?>
rhs: cc.Tree = 1
в отличие от:
scala> import reflect.runtime._, universe._
import reflect.runtime._
import universe._
scala> val DefDef(mods, nam, ps, vs, tpt, rhs) = q"final def x = 1"
mods: reflect.runtime.universe.Modifiers = Modifiers(final, , Map())
nam: reflect.runtime.universe.TermName = x
ps: List[reflect.runtime.universe.TypeDef] = List()
vs: List[List[reflect.runtime.universe.ValDef]] = List()
tpt: reflect.runtime.universe.Tree = <type ?>
rhs: reflect.runtime.universe.Tree = 1
Оказывается, что annotations
показывает только явные:
scala> mods.annotations
res13: List[reflect.runtime.universe.Tree] = List()
Стоит отметить, что док говорит, что showRaw
для проверки, а не для строительства.