3

我正在尝试将我的 java 代码移植到纯 scala,因此在这方面的任何帮助将不胜感激。

下面的代码首先将我的业务逻辑转换为 java 代码。在这里,我使用 freemarker 模板进行基于模板的代码生成。创建文件后,我使用 java 编译器编译代码并创建一个 jar 文件,该文件保存在一个临时目录中

我目前正在使用javax.tools.*提供运行时编译的包。与该方法等效的 Scala 是什么?我想使用 freemarker 模板生成纯 Scala 代码,然后运行 ​​Scala 编译来创建一个 jar 文件。

下面是我用来实现这一点的示例 Java 代码。

    JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
    Iterable<? extends JavaFileObject> compilationUnits = Arrays.asList(javaFileObjects);
    StringBuilder builder = new StringBuilder();
    builder.append(service.getConfig().getProp("coreLib"));
    builder.append(";" +result.getCodeContext().getOmClasspath());
    builder.append(";" +jarBuilder.toString());
    builder.append(";" +service.getConfig().getProp("tempCodeGen"));
    String[] compileOptions = new String[]{"-d", result.getCodeContext().getOmClasspath(),"-cp",builder.toString()} ;
    Iterable<String> compilationOptionss = Arrays.asList(compileOptions);
    DiagnosticCollector<JavaFileObject> diagnostics = new DiagnosticCollector<JavaFileObject>();
    CompilationTask compilerTask = compiler.getTask(null, stdFileManager, diagnostics, compilationOptionss, null, compilationUnits) ;
    boolean status = compilerTask.call();
4

4 回答 4

3

这是我自己的代码中的一些方法来编译项目并将其打包到 jar 中。它远未完善或正确评论,但希望它会指出您需要从哪里开始。我认为您不需要使用 String Builder,因为这不是性能关键:

def buildAll(name: String, projDir: String, mainClass: String = ""): Unit =
{
  import scala.tools.nsc.{Settings,Global}
  val relSrc : List[String] = List()
  val maniVersion = "None"
  def targDir: String = projDir + "/targ"
  def srcDir: String = projDir + "/src"
  def srcDirs: List[String] = srcDir :: relSrc

  import java.io._
  val sings = new scala.tools.nsc.Settings     
  new File(targDir).mkdir 
  sings.outputDirs.setSingleOutput(targDir.toString)     
  val comp = new Global(sings)      
  val crun: comp.Run  = new comp.Run
  def getList(fName: String): List[String] =
  {
     println("starting getList " + fName)
     val file = new File(fName)
     if (file.isDirectory) file.listFiles.toList.flatMap(i => getList(fName + "/" + i.getName))
     else List(fName)
  }  

  crun.compile(srcDirs.flatMap(i => getList(i))) 
  import sys.process._
  ("cp -r /sdat/projects/ScalaLibs/scala " + targDir + "/scala").!

  import java.util.jar._
  val manif = new Manifest
  val mf = manif.getMainAttributes
  mf.put(Attributes.Name.MANIFEST_VERSION, maniVersion)
  if (mainClass != "") mf.put(Attributes.Name.MAIN_CLASS, mainClass)
  val jarName = name + ".jar"
  val jarOut: JarOutputStream = new JarOutputStream(new FileOutputStream(projDir + "/" + jarName), manif)  
  AddAllToJar(targDir, jarOut)      
  jarOut.close   
}   

def addToJar(jarOut: JarOutputStream, file: File, reldir: String): Unit =
{
  val fName = reldir + file.getName         
  val fNameMod = if (file.isDirectory) fName + "/" else fName
  val entry = new JarEntry(fNameMod)
  entry.setTime(file.lastModified)         
  jarOut.putNextEntry(entry)
  if (file.isDirectory)
  {
     jarOut.closeEntry
     file.listFiles.foreach(i => addToJar(jarOut, i, fName + "/"))
  }
  else
  {         
     var buf = new Array[Byte](1024)
     val in = new FileInputStream(file)
     Stream.continually(in.read(buf)).takeWhile(_ != -1).foreach(jarOut.write(buf, 0, _))
     in.close
     jarOut.closeEntry()
  }         
}
def AddAllToJar(targDir: String, jarOut: JarOutputStream): Unit =
  new java.io.File(targDir).listFiles.foreach(i => addToJar(jarOut, i, ""))

您需要将 Scala 编译器添加到构建路径。Scala 编译器获取源文件列表并在输出目录中设置的目录中生成已编译的类文件。不过,掌握编译器的全部功能是一项主要任务。

于 2013-10-21T13:51:28.410 回答
1

而且在使用 scala 时,您不需要使用任何免费标记的类似工具。自 scala 2.10版以来,有字符串插值功能。

val traitName = "MyTrait"
val packageName = "my.pack"
val typeParams = List("A", "B", "C")

s"""
  |package ${packageName}
  |
  |trait ${traitName}[${typeParams.mkString(",")}] {
  |  ${typeParams.map(t => s"val ${t.toLowerCase()}: ${t}")}
  |}
  |
""".stripMargin

将产生:

package my.pack

trait MyTrait[A,B,C] {
  List(val a: A, val b: B, val c: C)
}

不需要依赖:)

于 2013-10-21T13:53:06.200 回答
1

如果您想在运行时编译生成的代码,最简单的解决方案是twitter eval 实用程序

于 2013-10-22T06:48:14.147 回答
0

请参阅此处的建议 -从字符串生成类并在 Scala 2.10 中实例化它

或者来自 twitter 的 util 库的 Eval 类 - https://github.com/twitter/util/blob/master/util-eval/src/main/scala/com/twitter/util/Eval.scala

于 2013-10-21T13:44:32.490 回答