在Gradle中分割所有输出目录

Gradle 4.0昨天出来,我更新了我的项目。

现在我收到以下警告:

Gradle现在为每个JVM语言使用单独的输出目录,但是这个版本假设源集合中所有类的单个目录。 这个行为已经被弃用,并计划在Gradle 5.0中被删除

我想为每种语言使用单独的输出目录。 为了做到这一点,我需要改变什么?


我试过的东西:

  • 随后由gradle build gradle clean
  • 删除构建目录,然后运行gradle build
  • 删除gradle和build目录然后运行gradle

相关的GitHub 问题


Gradle插件:

  • java的
  • 理念
  • org.springframework.boot

Gradle 4.0为每个JVM语言引入了多个sourceSet以启用远程构建缓存。 用java插件,你的build/classes/main应该成为build/classes/java/mainbuild/classes/test应该成为build/classes/java/test等。

该警告的定义如下:

https://github.com/gradle/gradle/blob/bb0ed850aa4061931a62133091d66c56551cf5e0/subprojects/plugins/src/main/java/org/gradle/api/internal/tasks/DefaultSourceSetOutput.java#L79

 @Override public File getClassesDir() { if (isLegacyLayout()) { return fileResolver.resolve(classesDir); } SingleMessageLogger.nagUserOfDeprecatedBehaviour("Gradle now uses separate output directories for each JVM language, but this build assumes a single directory for all classes from a source set"); Object firstClassesDir = CollectionUtils.findFirst(classesDirs.getFrom(), Specs.SATISFIES_ALL); if (firstClassesDir!=null) { return fileResolver.resolve(firstClassesDir); } return null; } 

所以,如果你的项目中有任何插件或者你的build.gradle调用DefaultSourceSetOutput.getClassesDir()你会得到这个警告。

正式推荐的方法来摆脱这个警告是

 sourceSets.main.output.classesDir = new File(buildDir, "classes/main") 

对应于:

 @Override public boolean isLegacyLayout() { return classesDir!=null; } @Override public void setClassesDir(File classesDir) { setClassesDir((Object)classesDir); } @Override public void setClassesDir(Object classesDir) { this.classesDir = classesDir; this.classesDirs.setFrom(classesDir); } 

请注意, https: getClassesDir() getClassesDir()标记为不建议使用

所以,直到所有在项目中使用的插件获得Gradle 4.0的支持,你应该坚持解决方法,并在Gradle源代码构建脚本中忽略弃用。

另一个问题是测试文件。 如果你不喜欢有不同的层次结构布局build/classes/mainbuild/classes/java/test ),你也应该调整测试路径:

 sourceSets.main.output.classesDir = new File(buildDir, "classes/main") sourceSets.test.output.classesDir = new File(buildDir, "classes/test") 

更新 IDEA的用户可能会注意到,IDE开始使用分离out目录来构建它检测到的Gradle 4.x。 如果您在IDEA之外运行应用程序,则会导致无法重新加载应用程序。 要解决这个添加和重新导入:

 subprojects { apply plugin: 'idea' // Due to Gradle 4.x changes (separate output directories per JVM language) // Idea developers refuse to reuse Gradle classpath and use own 'out/' directory. // Revert to old behavior to allow Spring Devtool to work with using fast Idea compiler. // https://youtrack.jetbrains.com/issue/IDEA-175172 // Alternatively use native Gradle builds or bootRun.addResources = true // To use this feature push Ctrl+Shift+F9 to recompile! // Be aware that Idea put resources into classes/ directory!! idea.module.inheritOutputDirs = false idea.module.outputDir = sourceSets.main.output.classesDir idea.module.testOutputDir = sourceSets.test.output.classesDir } 

请注意,IDEA把资源放到与.class文件相同的目录下,所以你的Gradle类路径将被损坏。 只需对您使用IDEA内置构建命令(Ctrl + Shift + F10等)的模块进行gradle clean

这是由于在Gradle 4.0中引入了更改:如果有多个语言源,它现在使用单独的输出目录。

要返回到旧的行为并摆脱警告,插入到您的build.gradle:

// Change the output directory for the main source set back to the old path sourceSets.main.output.classesDir = new File(buildDir, "classes/main")

参考: https : //docs.gradle.org/4.0/release-notes.html#multiple-class-directories-for-a-single-source-set

例如,如果你混合使用Java,Kotlin和Groovy项目结构应该如下所示:

 root/ src/ main/ java/ kotlin/ groovy/ test/ java/ kotlin/ groovy/ 

在你build.gradle你必须指定特定语言所需的插件。

 apply plugin: 'java' apply plugin: 'groovy' apply plugin: 'kotlin'