【问题标题】:How to compile and run Scala code programmatically如何以编程方式编译和运行 Scala 代码
【发布时间】:2019-09-16 21:56:29
【问题描述】:

我有以下代码,我想即时编译并运行它。

object HelloWorld {
  def main(args: Array[String]): Unit = {
    println("Hello, world!")
  }
}

到目前为止,我已经尝试过以下方法:

import scala.reflect.runtime.currentMirror
import scala.tools.reflect.ToolBox

object MainScala {
  def main(args: Array[String]): Unit = {
    val toolbox = currentMirror.mkToolBox()
    val source =
      """
        |object HelloWorld {
        |  def main(args: Array[String]): Unit = {
        |    println("Hello, world!")
        |  }
        |}
        |""".stripMargin
    val tree = toolbox.parse(source)
    val binary = toolbox.compile(tree)
    var c = binary.getClass
    val m = c.getMethod("main", classOf[Array[String]])
    val params = Array("Apple", "Banana", "Orange")
    m.invoke(null, null)
  }
}

toolbox.compile(tree) 之后我无法获取编译后代码的Class 对象。

【问题讨论】:

    标签: scala reflection scala-reflect scala-compiler


    【解决方案1】:

    如果您查看binary 的类型,您会发现它是() => Any。因此,当您询问.getClass 时,您实际上得到了Function1 的子类,它没有.main

    工具箱不应该这样使用。

    https://github.com/scala/scala/blob/2.13.x/src/compiler/scala/tools/reflect/ToolBox.scala#L120-L129

    https://docs.scala-lang.org/overviews/reflection/symbols-trees-types.html#tree-creation-via-parse-on-toolboxes

    试试

    val source =
      """
        |object HelloWorld {
        |  def main(args: Array[String]): Unit = {
        |    println("Hello, world!")
        |  }
        |}
        |
        |HelloWorld.main(Array())
        |""".stripMargin
    val tree = toolbox.parse(source)
    val binary = toolbox.compile(tree)
    binary() // Hello, world!
    

    val params = """Array("Apple", "Banana", "Orange")"""
    val source =
      s"""
        |object HelloWorld {
        |  def main(args: Array[String]): Unit = {
        |    println(args.toList)
        |  }
        |}
        |
        |HelloWorld.main($params)
        |""".stripMargin
    val tree = toolbox.parse(source)
    val binary = toolbox.compile(tree)
    binary() // List(Apple, Banana, Orange)
    

    import scala.reflect.runtime.universe._
    val params = q"""Array("Apple", "Banana", "Orange")"""
    val tree =
      q"""
        object HelloWorld {
          def main(args: Array[String]): Unit = {
            println(args.toList)
          }
        }
    
        HelloWorld.main($params)
        """
    val binary = toolbox.compile(tree)
    binary() // List(Apple, Banana, Orange)
    

    val params = """Array("Apple", "Banana", "Orange")"""
    val source =
      """
        |object HelloWorld {
        |  def main(args: Array[String]): Unit = {
        |    println(args.toList)
        |  }
        |}
        |""".stripMargin
    val tree = toolbox.parse(source)
    val symbol = toolbox.define(tree.asInstanceOf[ImplDef])
    val params1 = toolbox.parse(params)
    val tree1 = q"$symbol.main($params1)"
    val binary = toolbox.compile(tree1)
    binary() // List(Apple, Banana, Orange)
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2012-05-06
      • 1970-01-01
      • 1970-01-01
      • 2022-07-15
      • 1970-01-01
      • 2015-03-12
      • 1970-01-01
      • 2017-10-31
      相关资源
      最近更新 更多