在我们的项目中,我们对编译生成的.class文件进行了增强的后处理。这个增强步骤实际上修改了生成的.class文件,然后覆盖它.
enhance <<= enhance triggeredBy (compile in Compile)
问题是sbt有一种称为增量重新编译的机制。它监视生成的.class文件。每次增强器重写生成的.class文件时,sbt都会识别这些修改,并在下一个编译命令中重新编译相关的源代码。
对我们来说,重新编译是一项非常耗时的工作。我们希望阻止sbt重新编译修改后的.class文件。这可能意味着sbt只监视源更改,而不监视输出更改。
我对此做了一些搜索。但我发现了一些小事情。现在我知道一个叫做Analysis的特性可能负责从源文件到输出.class文件的映射。所以我请求你们的帮助。
Ps:我们可以通过将enhance的输出放到另一个文件夹来解决这个问题,但这不是首选。
sbt强烈反对对文件进行突变。您应该生成不同的文件。通过这样做,您将解决问题,因为sbt的增量编译器仍将查看未更改的.class文件。你需要重新布线:
将compile
任务的输出发送到其他地方:
classDirectory in Compile := crossTarget.value / "classes-orig"
使用工具处理这些.class文件,并将它们发送到crossTarget.value / "classes"
(原始classDirectory
:
enhance <<= enhance triggeredBy (compile in Compile)
enhance := {
val fromDir := (classesDirectory in Compile).value
val toDir := crossTarget.value / "classes"
...
}
无论如何,重新连接productDirectories
以使用crossTarget.value / "classes"
(否则它将在修改后的classDirectory
:中查找
productDirectories in Compile := Seq(crossTarget.value / "classes")
确保products
取决于您的enhance
任务:
products in Compile <<= (products in Compile) dependsOn enhance
如果您有资源,您可能需要重新布线(请参阅copyResources
)。但基本上你应该能够到达那里。
我说过sbt监视output.class文件。当.class文件被修改时,它会重新编译.class的源文件。
经过研究,我们发现sbt通过其上次修改时间来通知文件的修改。也就是说,我们可以通过在修改后回滚上次修改的时间来欺骗sbt,这样sbt就不会注意到任何变化。
因此,我们的解决方案简单但有效:
- 查找所有.class文件
- 记下他们最后修改的时间
- 增强
- 放回以前最后修改的时间
这是个小把戏。我们仍然期待更稳健的解决方案。
描述:
有点像Chenyu,我必须为SBT 1.x
编写一个插件,该插件将增强已编译的类。后来我想确保这些的增强类用于构建jar
。
我不想破解这个解决方案,所以陈宇的回答对我来说是不可接受的,sjrd的回答很有帮助,但调整为SBT 0.13
。
以下是我的工作解决方案,只需少量评论:
代码:
object autoImport {
val enhancedDest = settingKey[File]("Output dir for enhanced sources")
}
def enhanceTask: Def.Initialize[Task[Unit]] = Def.task {
val inputDir = (classDirectory in Compile).value
val outputDir = enhancedDest.value
enhance(inputDir, outputDir)
...
}
override def projectSettings: Seq[Def.Setting[_]] = Seq(
enhancedDest := crossTarget.value / "classes-enhanced",
products in Compile := Seq(enhancedDest.value), // mark enhanced folder to use for packaging
// https://www.scala-sbt.org/1.0/docs/Howto-Dynamic-Task.html#build.sbt+v2
compile in Compile := Def.taskDyn {
val c = (compile in Compile).value // compile 1st.
Def.task {
(copyResources in Compile).value // copy resources before enhance
enhanceTask.value // enhance
c
}
}.value
)