1

我正在尝试编写一个简单的程序来遍历从给定方法开始的所有引用代码,使用scalameta.

我能够跟踪调用,但无法解析方法引用。

分析/src/main/scala/codelab/FindMe.scala

package codelab

object FindMe {
  def main(args: Array[String]): Unit = {
    val x = someRecognizeableName(1, 2)
    val y = List(1, 2, 3)
    y.foldLeft(0)(someRecognizeableName)
  }
  def someRecognizeableName(a: Int, b: Int): Int = a + b
}

生成并加载语义数据库FindMe.scala并检查方法的用法someRecognizeableName

我可以看到db.names列表中的第一个调用:

[87..108): someRecognizeableName => _root_.codelab.FindMe.someRecognizeableName(Int,Int).

但是,第二个,当我不调用该方法时,只需传递引用显示如下:

[159..180): someRecognizeableName => local2_src_main_scala_codelab_FindMe_scala

因此,当我尝试从 开始跟踪引用时,在第二种情况下main,我没有得到引用的完全限定名称。someRecognizeableName

问题:有没有办法从语义数据库中为该引用获取一个完全限定的名称?



重现上述内容的完整来源

运行说明:

analyzeme $ sbt compile
analyzer $ sbt "run ../analyzeme"

分析/src/main/scala/codelab/FindMe.scala

package codelab

object FindMe {
  def main(args: Array[String]): Unit = {
    val x = someRecognizeableName(1, 2)
    val y = List(1, 2, 3)
    y.foldLeft(0)(someRecognizeableName)
  }
  def someRecognizeableName(a: Int, b: Int): Int = a + b
}

分析器/src/main/scala/Main.scala

import org.langmeta.io.{Classpath, Sourcepath}

import scala.meta._

object Main {
  def main(args: Array[String]): Unit = {

    println(s"Loading from [${ args(0) }]")
    println()

    val cp = Classpath(s"${ args(0) }/target/scala-2.12/classes")
    val sp = Sourcepath(s"${ args(0) }/src/main/scala")

    val db = Database.load(cp, sp)

    println("* names:")
    db.names foreach println
    println()
    println("* symbols:")
    db.symbols foreach println
    println()
    println("* synthetics:")
    db.synthetics foreach println
    println()
    println("* messages:")
    db.messages foreach println
    println()
  }

}

分析/构建.sbt

name := "analyzee"
version := "0.1"
scalaVersion := "2.12.4"

addCompilerPlugin("org.scalameta" % "semanticdb-scalac" % "3.4.0" cross CrossVersion.full)
scalacOptions += "-Yrangepos"

分析器/build.sbt

name := "analyzer"
version := "0.1"
scalaVersion := "2.12.4"

libraryDependencies += "org.scalameta" %% "scalameta" % "3.4.0"
libraryDependencies += "org.scalameta" %% "contrib" % "3.4.0"
4

1 回答 1

1
package codelab

object FindMe {
  def main(args: Array[String]): Unit = {
  val x = someRecognizeableName(1, 2)

  y.foldLeft(0)(someRecognizeableName)
  // same as
  y.foldLeft(0){ a, b => someRecognizeableName(a, b) }
}

我调试代码并在第二种情况下发现,编译器传递了一个无法从当前语义数据库访问的匿名符号,它可能应该在 syhthetics 部分中,但我在里面找不到它。

所以我猜当前语义数据库中缺少编译器生成的匿名。

于 2018-03-08T11:30:28.567 回答