- 921. Minimum Add to Make Parentheses Valid 使括号有效的最少添加
- 915. Partition Array into Disjoint Intervals 分割数组
- 932. Beautiful Array 漂亮数组
- 940. Distinct Subsequences II 不同的子序列 II
Java 插件向一个项目添加了 Java 编译、 测试和 bundling 的能力。它是很多其他 Gradle 插件的基础服务。
要使用Java 插件,请在构建脚本中加入:
使用 Java 插件
build.gradle
apply plugin: 'java'
Java 插件引入了一个源集的概念。一个源集只是一组用于编译并一起执行的源文件。这些源文件可能包括 Java 源代码文件和资源文件。其他有一些插件添加了在源集里包含 Groovy 和 Scala 的源代码文件的能力。一个源集有一个相关联的编译类路径和运行时类路径。
源集的一个用途是,把源文件进行逻辑上的分组,以描述它们的目的。例如,你可能会使用一个源集来定义一个集成测试套件,或者你可能会使用单独的源集来定义你的项目的 API 和实现类。
Java 插件定义了两个标准的源集,分别是 main 和 test。main 源集包含你产品的源代码,它们将被编译并组装成一个 JAR 文件。test 源集包含你的单元测试的源代码,它们将被编译并使用 JUnit 或 TestNG 来执行。
Java 插件向你的项目添加了大量的任务,如下所示。
表23.1. Java 插件-任务
任务名称 | 依赖于 | 类型 | 描述 |
compileJava |
产生编译类路径中的所有任务。这包括了用于jar 任务。 |
JavaCompile | 使用 javac 编译产品中的 Java 源文件。 |
processResources |
– | Copy | 把生产资源文件拷贝到生产的类目录中。 |
classes |
processResources 。一些插件添加了额外的编译任务。 |
Task | 组装生产的类目录。 |
compileTestJava |
compile ,再加上所有能产生测试编译类路径的任务。 |
JavaCompile | 使用 javac 编译 Java 的测试源文件。 |
processTestResources |
– | Copy | 把测试的资源文件拷贝到测试的类目录中。 |
testClasses |
processTestResources 。一些插件添加了额外的测试编译任务。 |
Task | 组装测试的类目录。 |
jar |
compile |
Jar | 组装 JAR 文件 |
javadoc |
compile |
Javadoc | 使用 Javadoc 生成生产的 Java 源代码的API文档 |
test |
compileTest ,再加上所有产生测试运行时类路径的任务。 |
Test | 使用 JUnit 或 TestNG运行单元测试。 |
uploadArchives |
使用jar 。 |
Upload | 使用archives 配置上传包括 JAR 文件的构件。 |
clean |
– | Delete | 删除项目的 build 目录。 |
TaskName |
– | Delete | 删除由指定的任务所产生的输出文件。例如, jar 任务中所创建的 JAR 文件,test 任务所创建的测试结果。 |
对于每个你添加到该项目中的源集,Java 插件将添加以下的编译任务:
表23.2. Java 插件-源集任务
任务名称 | 依赖于 | 类型 | 描述 |
SourceSet Java |
所有产生源集编译类路径的任务。 | JavaCompile | 使用 javac 编译给定的源集中的 Java 源文件。 |
SourceSet Resources |
– | Copy | 把给定的源集的资源文件拷贝到类目录中。 |
sourceSet Classes |
SourceSet Resources。某些插件还为源集添加了额外的编译任务。 |
Task | 组装给定源集的类目录。 |
Java 插件还增加了大量的任务构成该项目的生命周期:
表23.3. Java 插件-生命周期任务
任务名称 | 依赖于 | 类型 | 描述 |
assemble |
项目中的所有归档项目,包括jar 任务。某些插件还向项目添加额外的归档任务。 |
Task | 组装项目中所有的归类文件。 |
check |
项目中的所有核查项目,包括test 任务。某些插件还向项目添加额外的核查任务。 |
Task | 执行项目中所有的核查任务。 |
build |
assemble |
Task | 执行项目的完事构建。 |
buildNeeded |
build 任务。 |
Task | 执行项目本身及它所依赖的其他所有项目的完整构建。 |
buildDependents |
build 任务。 |
Task | 执行项目本身及依赖它的其他所有项目的完整构建。 |
ConfigurationName |
使用配置ConfigurationName生成构件的任务。 | Task | 组装指定配置的构件。该任务由Base插件添加,并由Java插件隐式实现。 |
ConfigurationName |
使用配置ConfigurationName上传构件的任务。 | Upload | 组装并上传指定配置的构件。该任务由Base插件添加,并由Java插件隐式实现。 |
uploadConfigurationName 使用配置 ConfigurationName 上传构件的任务。 Upload 组装并上传指定配置的构件。该任务由 Base 插件添加,并由 Java 插件隐式实现。 下图显示了这些任务之间的关系。
图23、1. Java 插件 – 任务
Java 插件 – 任务
Java 插件会假定如下所示的项目布局。这些目录都不需要一定存在,或者是里面有什么内容。Java 插件将会进行编译,不管它发现什么,并处理缺少的任何东西。
表23.4. Java 插件-默认项目布局
目录 | 意义 |
src/main/java |
产品的Java源代码 |
src/main/resources |
产品的资源 |
src/test/java |
Java 测试源代码 |
src/test/resources |
测试资源 |
sourceSet /java |
给定的源集的Java源代码 |
sourceSet /resources |
给定的源集的资源 |
你可以通过配置适当的源集,来配置项目的布局。这一点将在以下各节中详细讨论。这里是如何更改 main Java 和资源源目录的一个简短的例子。
自定义 Java 源代码布局
build.gradle
sourceSets {
main {
java {
srcDir 'src/java'
}
resources {
srcDir 'src/resources'
}
}
}
Java 插件向项目添加了许多依赖配置,如下图所示。它对一些任务指定了这些配置,如 compileJava 和 test。
表23、5. Java插件 – 依赖配置
名称 | 继承自 | 在哪些任务中使用 | 意义 |
compile | – | compileJava | 编译时依赖 |
runtime | compile | – | 运行时依赖 |
testCompile | compile | compileTestJava | 用于编译测试的其他依赖 |
testRuntime | runtime, testCompile | test | 只用于运行测试的其他依赖 |
archives | – | uploadArchives | 由本项目生产的构件(如jar包)。 |
default | runtime | – | 本项目上的默认项目依赖配置。包含本项目运行时所需要的构件和依赖。 |
图23、2. Java 插件 – 依赖配置
Java 插件 – 依赖配置
对于每个你添加到项目中的源集,Java 插件都会添加以下的依赖配置:
表23、6. Java 插件 – 源集依赖配置
名称 | 继承自 | 在哪些任务中使用 | 意义 |
sourceSetCompile | – | compileSourceSetJava | 给定源集的编译时依赖 |
sourceSetRuntime | sourceSetCompile | – | 给定源集的运行时依赖 |
Java 插件向项目添加了许多常规属性,如下图所示。您可以在构建脚本中使用这些属性,就像它们是 project 对象的属性一样。
表23、7. Java 插件 – 目录属性
属性名称 | 类型 | 默认值 | 描述 |
reportsDirName |
String |
reports |
相对于build目录的目录名称,报告将生成到此目录。 |
reportsDir |
File (read-only) |
reportsDirName |
报告将生成到此目录。 |
testResultsDirName |
String |
test-results |
相对于build目录的目录名称,测试报告的.xml文件将生成到此目录。 |
testResultsDir |
File (read-only) |
testResultsDirName |
测试报告的.xml文件将生成到此目录。 |
testReportDirName |
String |
tests |
相对于build目录的目录名称,测试报告将生成到此目录。 |
testReportDir |
File (read-only) |
testReportDirName |
测试报告生成到此目录。 |
libsDirName |
String |
libs |
相对于build目录的目录名称,类库将生成到此目录中。 |
libsDir |
File (read-only) |
libsDirName |
类库将生成到此目录中。 |
distsDirName |
String |
distributions |
相对于build目录的目录名称,发布的文件将生成到此目录中。 |
distsDir |
File (read-only) |
distsDirName |
要发布的文件将生成到此目录。 |
docsDirName |
String |
docs |
相对于build目录的目录名称,文档将生成到此目录。 |
docsDir |
File (read-only) |
docsDirName |
要生成文档的目录。 |
dependencyCacheDirName |
String |
dependency-cache |
相对于build目录的目录名称,该目录用于缓存源代码的依赖信息。 |
dependencyCacheDir |
File (read-only) |
dependencyCacheDirName |
该目录用于缓存源代码的依赖信息。 |
表23.8. Java 插件 – 其他属性
属性名称 | 类型 | 默认值 | 描述 |
sourceSets |
SourceSetContainer (read-only) | 非空 | 包含项目的源集。 |
sourceCompatibility |
JavaVersion. 可以使用字符串或数字来设置,例如1.5 。 |
当前JVM所使用的值 | 当编译Java源代码时所使用的Java版本兼容性。 |
targetCompatibility |
JavaVersion. 可以使用字符串或数字来设置,例如1.5 。 |
sourceCompatibility |
要生成的类的 Java 版本。 |
archivesBaseName |
String |
projectName |
像JAR或ZIP文件这样的构件的basename |
manifest |
Manifest | 一个空的清单 | 要包括的所有 JAR 文件的清单。 |
这些属性由 JavaPluginConvention, BasePluginConvention 和 ReportingBasePluginConvention 这些类型的常规对象提供。
你可以使用 sourceSets 属性访问项目的源集。这是项目的源集的容器,它的类型是 SourceSetContainer。除此之后,还有一个 sourceSets {}的脚本块,可以传入一个闭包来配置源集容器。源集容器的使用方式几乎与其他容器一样,例如 tasks。
访问源集
build.gradle
// Various ways to access the main source set
println sourceSets.main.output.classesDir
println sourceSets['main'].output.classesDir
sourceSets {
println main.output.classesDir
}
sourceSets {
main {
println output.classesDir
}
}
// Iterate over the source sets
sourceSets.all {
println name
}
要配置一个现有的源集,你只需使用上面的其中一种访问方法来设置源集的属性。这些属性将在下文中进行介绍。下面是一个配置 main 的 Java 和资源目录的例子:
配置源集的源代码目录
build.gradle
sourceSets {
main {
java {
srcDir 'src/java'
}
resources {
srcDir 'src/resources'
}
}
}
下表列出了一些重要的源集属性。你可以在 SourceSet 的 API 文档中查看更多的详细信息。
表23.9. Java 插件 – 源集属性
属性名称 | 类型 | 默认值 | 描述 |
name |
String (read-only) |
非空 | 用来确定一个源集的源集名称。 |
output |
SourceSetOutput (read-only) | 非空 | 源集的输出文件,包含它编译过的类和资源。 |
output.classesDir |
File |
name |
要生成的该源集的类的目录。 |
output.resourcesDir |
File |
name |
要生成的该源集的资源的目录。 |
compileClasspath |
FileCollection | SourceSet 配置。 |
该类路径在编译该源集的源文件时使用。 |
runtimeClasspath |
FileCollection | SourceSet 配置。 |
该类路径在执行该源集的类时使用。 |
java |
SourceDirectorySet (read-only) | 非空 | 该源集的Java源文件。仅包含Java源文件目录里的.java 文件,并排除其他所有文件。 |
java.srcDirs |
Set<File> |
name /java] |
该源目录包含了此源集的所有Java源文件。 |
resources |
SourceDirectorySet (read-only) | 非空 | 此源集的资源文件。仅包含资源文件,并且排除在资源源目录中找到的所有 .java 文件。其他插件,如Groovy 插件,会从该集合中排除其他类型的文件。 |
resources.srcDirs |
Set<File> |
name /resources] |
该源目录包含了此源集的资源文件。 |
allJava |
SourceDirectorySet (read-only) | java |
该源集的所有.java 文件。有些插件,如Groovy 插件,会从该集合中增加其他的Java源文件。 |
allSource |
SourceDirectorySet (read-only) | resources + java |
该源集的所有源文件。包含所有的资源文件和Java源文件。有些插件,如Groovy 插件,会从该集合中增加其他的源文件。 |
要定义一个新的源集,你只需在 sourceSets {}块中引用它。下面是一个示例:
定义一个源集
build.gradle
sourceSets {
intTest
}
当您定义一个新的源集时,Java 插件会为该源集添加一些依赖配置,如表 23.6,“Java 插件 – 源集依赖项配置”所示。你可以使用这些配置来定义源集的编译和运行时的依赖。
定义源集依赖
build.gradle
sourceSets {
intTest
}
dependencies {
intTestCompile 'junit:junit:4.11'
intTestRuntime 'org.ow2.asm:asm-all:4.0'
}
Java 插件还添加了大量的任务,用于组装源集的类,如表 23.2,“Java 插件 – 源设置任务”所示。例如,对于一个被叫做 intTest 的源集,你可以运行 gradle intTestClasses 来编译 int 测试类。
编译源集
gradle intTestClasses的输出结果
> gradle intTestClasses
:compileIntTestJava
:processIntTestResources
:intTestClasses
BUILD SUCCESSFUL
Total time: 1 secs
添加一个包含了源集的类的 JAR 包
示例 23.8. 为一个源集装配一个JAR文件
build.gradle
task intTestJar(type: Jar) {
from sourceSets.intTest.output
}
为一个源集生成 Javadoc:
示例 23.9. 为一个源集生成 Javadoc:
build.gradle
task intTestJavadoc(type: Javadoc) {
source sourceSets.intTest.allJava
}
添加一个测试套件以运行一个源集里的测试
示例 23.10. 运行源集里的测试
build.gradle
task intTest(type: Test) {
testClassesDir = sourceSets.intTest.output.classesDir
classpath = sourceSets.intTest.runtimeClasspath
}
Javadoc 任务是 Javadoc 的一个实例。它支持核心的 javadoc 参数选项,以及在 Javadoc 可执行文件的参考文档中描述的标准 doclet 参数选项。对于支持的 Javadoc 参数选项的完整列表,请参考下面的类的 API 文档: CoreJavadocOptions 和StandardJavadocDocletOptions。
表23.10. Java 插件 – Javadoc 属性
任务属性 | 类型 | 默认值 |
classpath |
FileCollection | sourceSets.main.output + sourceSets.main.compileClasspath |
source |
FileTree. | sourceSets.main.allJava |
destinationDir |
File |
docsDir /javadoc |
title |
String |
project的名称和版本 |
clean 任务是 Delete 的一个实例。它只是删除由其 dir 属性表示的目录。
表23.11. Java 插件 – Clean 性能
任务属性 | 类型 | 默认值 |
dir |
File |
buildDir |
Java 插件使用 Copy 任务进行资源的处理。它为该 project 里的每个源集添加一个实例。你可以在16.6章节,“复制文件”中找到关于 copy 任务的更多信息。
表23.12. Java 插件-ProcessResources 属性
任务属性 | 类型 | 默认值 |
srcDirs |
Object . |
sourceSet .resources |
destinationDir |
16.1 章节,“查找文件”中所讲到的任何一种方式来设置。 |
sourceSet .output.resourcesDir |
Java 插件为该 project 里的每个源集添加一个 JavaCompile 实例。一些最常见的配置选项如下所示。
表23.13. Java 插件- Compile 属性
任务属性 | 类型 | 默认值 |
classpath |
FileCollection | sourceSet .compileClasspath |
source |
FileTree | sourceSet .java |
destinationDir |
File . |
sourceSet .output.classesDir |
compile 任务委派了 Ant 的 javac 任务。将 options.useAnt 设置为 false 将绕过 Ant 任务,而激活 Gradle 的直接编译器集成。在未来的 Gradle 版本中,将把它作为默认设置。
默认情况下,Java 编译器在 Gradle 过程中运行。将 options.fork 设置为 true 将会使编译出现在一个单独的进程中。在 Ant javac 任务中,这意味着将会为每一个 compile 任务fork 一个新的进程,而这将会使编译变慢。相反,Gradle 的直接编译器集成 (见上文) 将尽可能多地重用相同的编译器进程。在这两种情况下,使用 options.forkOptions 指定的所有fork 选项都将得到重视。
test 任务是 Test 的一个实例。它会自动检测和执行 test 源集中的所有单元测试。测试执行完成后,它还会生成一份报告。同时支持 JUnit 和 TestNG。可以看一看Test的完整的 API。
测试在单独的 JVM 中执行,与 main 构建进程隔离。Test 任务的 API 可以让你控制什么时候开始。
有大量的属性用于控制测试进程的启动。这包括系统属性、 JVM 参数和使用的 Java 可执行文件。
你可以指定是否要并行运行你的测试。Gradle 通过同时运行多个测试进程来提供并行测试的执行。每个测试进程会依次执行一个单一的测试,所以你一般不需要对你的测试做任何的配置来利用这一点。 MaxParallelForks 属性指定在给定的时间可以运行的测试进程的最大数。它的默认值是 1,也就是说,不并行执行测试。
测试进程程会为其将 org.gradle.test.worker 系统属性设置为一个唯一标识符,这个标识符可以用于文件名称或其他资源标识符。
你可以指定在执行了一定数量的测试类之后,重启那个测试进程。这是一个很有用的替代方案,让你的测试进程可以有很大的堆内存。forkEvery 属性指定了要在测试进程中执行的测试类的最大数。默认是每个测试进程中执行的测试数量不限。
该任务有一个 ignoreFailures 属性,用以控制测试失败时的行为。test 会始终执行它检测到的每一个测试。如果 ignoreFailures 为 false,并且有测试不通过,那它会停止继续构建。IgnoreFailures 的默认值为 false。
testLogging 属性可以配置哪些测试事件需要记录,并且使用什么样的日志级别。默认情况下,对于每个失败的测试都只会打印一个简洁的消息。请参阅 TestLoggingContainer,查看如何把你的测试日志打印调整为你的偏好设置。
test 任务提供了一个 Test.getDebug()属性,可以设置为启动,使 JVM 在执行测试之前,等待一个 debugger 连接到它的 5005 端口上。
这也可以在调用时通过–debug-vm task 选项进行启用。
从Gradle 1.10 开始,可以根据测试的名称模式,只包含指定的测试。过滤,相对于测试类的包含或排除,是一个不同的机制。它将在接下来的段落中进行描述(-Dtest.single, test.include 和 friends)。后者基于文件,比如测试实现类的物理位置。file-level 的测试选择不支持的很多有趣的情况,都可以用 test-level 过滤来做到。以下这些场景中,有一些 Gradle 现在就可以处理,而有一些则将在未来得到实现:
测试过滤功能具有以下的特征:
在构建脚本中过滤测试
build.gradle
test {
filter {
//include specific method in any of the tests
includeTestsMatching "*UiCheck"
//include all tests from package
includeTestsMatching "org.gradle.internal.*"
//include all integration tests
includeTestsMatching "*IntegTest"
}
}
有关更多的详细信息和示例,请参阅 TestFilter 的文档。
使用命令行选项的一些示例:
这种机制已经被上述的“测试过滤”所取代。 设置一个 taskName.single = testNamePattern 的系统属性将会只执行匹配 testNamePattern 的那些测试。这个 taskName 可以是一个完整的多项目路径,比如“sub1:sub2:test”,或者仅是一个任务名称。testNamePattern 将用于形成一个“*/testNamePattern.class” 的包含模式。如果这个模式无法找到任何测试,那么将会抛出一个异常。这是为了使你不会误以为测试通过。如果执行了一个以上的子项目的测试,该模式会被应用于每一个子项目中。如果在一个特定的子项目中,找不到测试用例,那么将会抛出异常。在这种情况下你可以使用路径标记法的模式,这样该模式就会只应用于特定的子项目的测试任务中。或者你可以指定要执行的任务的完整限定名称。你还可以指定多个模式。示例:
Test 任务通过检查编译过的测试类来检测哪些类是测试类。默认情况下它会扫描所有的.class 文件。您可以设置自定义的 includes 或 excludes,这样就只有这些类才会被扫描。根据所使用的测试框架 (JUnit 或 TestNG),会使用不同的标准进行测试类的检测。
当使用JUnit 时,我们扫描 JUnit 3 和 4 的测试类。如果满足以下的任何一个条件,这个类就被认为是一个 JUnit 测试类:
当使用TestNG 时,我们扫描所有带有 @Test 注解的方法。
请注意,抽象类不会被执行。Gradle 还将扫描测试类路径中的 jar 文件里的继承树。
如果你不想要使用测试类检测,可以通过设置 scanForTestClasses 为 false 来禁用它。这将使test任务只使用 includes / excludes 来找到测试类。如果 scanForTestClasses 为disabled,并且没有指定 include 或 exclude 模式,则使用各自的默认值。对于 include 的默认值是 “/*Tests.class”, “**/*Test.class”,而对于exclude它的默认值是 “/Abstract*.class”。
JUnit 和 TestNG 可以对测试方法进行复杂的分组。
为对Junit 测试类和方法进行分组,JUnit 4.8 引入了类别的概念。test 任务可以实现一个规范,让你 include 和 exclude 想要的 JUnit 类别。
JUnit 类别
build.gradle
test {
useJUnit {
includeCategories 'org.gradle.junit.CategoryA'
excludeCategories 'org.gradle.junit.CategoryB'
}
}
TestNG 框架有一个非常相似的概念。在 TestNG 中你可以指定不同的测试组。应从测试执行中 include 或 exclude 的测试组,可以在 test 任务中配置。
对TestNG 测试分组
build.gradle
test {
useTestNG {
excludeGroups 'integrationTests'
includeGroups 'unitTests'
}
}
test 任务默认情况下会生成以下结果。
您可以使用 Test.setTestReport()方法来禁用 HTML 测试报告。目前不能禁用其他的结果。
这里还有一个独立的 TestReport 任务类型,它可以从一个或多个 Test 任务实例生成的二进制结果中生成 HTML 测试报告。要使用这个任务类型,你需要定义一个 destinationDir 和要包含到报告的测试结果。这里是一个范例,从子项目的单元测试中生成一个联合报告:
为多个子项目创建一个单元测试报告
build.gradle
subprojects {
apply plugin: 'java'
// Disable the test report for the individual test task
test {
reports.html.enabled = false
}
}
task testReport(type: TestReport) {
destinationDir = file("$buildDir/reports/allTests")
// Include the results from the test task in all subprojects
reportOn subprojects*.test
}
你应该注意到,TestReport 类型组合了多个测试任务的结果,并且需要聚合各个测试类的结果。这意味着,如果一个给定的测试类被多个 test 任务所执行,那么测试报告将包括那个类的所有执行结果,但它难以区分那个类的每次执行和它们的输出。
TestNG 支持参数化测试方法,允许一个特定的测试方法使用不同的输入执行多次。Gradle 会在这个方法的执行报告中包含进它的参数值。
给定一个带有两个参数,名为 aParameterizedTestMethod 参数化测试方法,它将使用名称这个名称进行报告 :aParameterizedTestMethod(toStringValueOfParam1, toStringValueOfParam2)。这使得在特定的迭代过程,很容易确定参数值。
表23.14. Java 插件 – test 属性
任务属性 | 类型 | 默认值 |
testClassesDir |
File |
sourceSets.test.output.classesDir |
classpath |
FileCollection | sourceSets.test.runtimeClasspath |
testResultsDir |
File |
testResultsDir |
testReportDir |
File |
testReportDir |
testSrcDirs |
List |
sourceSets.test.java.srcDirs |
Jar任务创建一个包含类文件和项目资源的 JAR 文件。JAR 文件在 archives 依赖配置中被声明为一个构件。这意味着这个 JAR 文件被包含在一个依赖它的项目的类路径中。如果你把你的项目上传到仓库,这个 JAR 文件会被声明为依赖描述符的一部分。你可以在第16.8节,“创建档案”和第51章, 发布 artifact 中了解如何使用 archives 和配置 artifact。
每个jar 或 war 对象都有一个单独的 Manifest 实例的 manifest 属性。当生成 archive 时,相应的 MANIFEST.MF 文件也会被写入进去。
自定义的 MANIFEST.MF
build.gradle
jar {
manifest {
attributes("Implementation-Title": "Gradle", "Implementation-Version": version)
}
}
您可以创建一个单独的 Manifest 实例。它可以用于共享两个 jar 包的 manifest 信息。
创建一个 manifest 对象。
build.gradle
ext.sharedManifest = manifest {
attributes("Implementation-Title": "Gradle", "Implementation-Version": version)
}
task fooJar(type: Jar) {
manifest = project.manifest {
from sharedManifest
}
}
你可以把其他的 manifest 合并到任何一个 Manifest 对象中。其他的 manifest 可能使用文件路径来描述,像上面的例子,使用对另一个 Manifest 对象的引用。
指定 archive 的单独 MANIFEST.MF
build.gradle
task barJar(type: Jar) {
manifest {
attributes key1: 'value1'
from sharedManifest, 'src/config/basemanifest.txt'
from('src/config/javabasemanifest.txt', 'src/config/libbasemanifest.txt') {
eachEntry { details ->
if (details.baseValue != details.mergeValue) {
details.value = baseValue
}
if (details.key == 'foo') {
details.exclude()
}
}
}
}
}
Manifest 会根据在 from 语句中所声明的顺序进行合并。如果基本的 manifest 和要合并的 manifest 都定义了同一个 key 的值,那么默认情况下会采用要合并的 manifest 的值。你可以通过添加 eachEntry action 来完全自定义合并行为,它可以让你对每一项生成的 manifest 访问它的一个 ManifestMergeDetails 实例。这个合并操作不会在 from 语句中就马上被触发。它是懒执行的,不管是对于生成一个 jar 包,还是调用了 writeTo 或者 effectiveManifest
你可以很轻松地把一个 manifest 写入磁盘中。
指定 archive 的单独 MANIFEST.MF
build.gradle
jar.manifest.writeTo("$buildDir/mymanifest.mf")
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!