这篇教程的主要内容是讲解如何用Gradle编译和打包一个简单的Java项目。
该Java项目只有一个需求:我们的构建脚本必须创建一个可执行的Jar文件,换句话说,我们必须能够使用命令java -jar jarfile.jar 来运行我们的程序。我们来看一下如何满足这个需求。
我们可以使用Java插件(译注:关于Gradle插件的定义,请查看第一篇教程)来创建一个Java项目,为了做到这点,我们需要把下面这段语句加入到build.gradle文件中:
1
|
apply plugin:
'java'
|
就是这样,现在我们已经创建了一个Java项目。Java插件会在我们的构建中添加一些新的约定(如默认的项目结构),新的任务,和新的属性。
让我们来快速地看一下默认的项目结构。
默认的项目结构如下:
让我们创建一个简单的主类,在这个类中会打印一个“Hello world”然后System.out出来。这个HelloWorld类的源代码如下:
1
2
3
4
5
6
7
8
|
package
net.petrikainulainen.gradle;
public
class
HelloWorld {
public
static
void
main(String[] args) {
System.out.println(
"Hello World!"
);
}
}
|
HelloWorld类存放在src/main/java/net/petrikainulainen/gradle目录
这很好,然而,我们还需要编译和打包我们的项目,不是吗?我们先看一下这个Java工程中的任务。
Java插件在我们的构建中加入了很多任务,我们这篇教程涉及到的任务如下:
我们还可以执行以下命令得到一个可运行任务及其描述的完整列表
1
|
gradle tasks
|
这是一个很好的方式,不需要阅读构建脚本,就能对你的项目进行大致的浏览,如果我们在项目根目录下运行这个命令,我们可以看到以下输出:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
|
> gradle tasks
:tasks
------------------------------------------------------------
All tasks runnable from root project
------------------------------------------------------------
Build tasks
-----------
assemble - Assembles the outputs of this project.
build - Assembles and tests this project.
buildDependents - Assembles and tests this project and all projects that depend on it.
buildNeeded - Assembles and tests this project and all projects it depends on.
classes - Assembles classes
'main'
.
clean - Deletes the build directory.
jar - Assembles a jar archive containing the main classes.
testClasses - Assembles classes
'test'
.
Build Setup tasks
-----------------
init - Initializes a new Gradle build. [incubating]
wrapper - Generates Gradle wrapper files. [incubating]
Documentation tasks
-------------------
javadoc - Generates Javadoc API documentation
for
the main
source
code.
Help tasks
----------
dependencies - Displays all dependencies declared
in
root project
'first-java-project'
.
dependencyInsight - Displays the insight into a specific dependency
in
root project
'first-java-project'
.
help - Displays a help message
projects - Displays the sub-projects of root project
'first-java-project'
.
properties - Displays the properties of root project
'first-java-project'
.
tasks - Displays the tasks runnable from root project
'first-java-project'
.
Verification tasks
------------------
check - Runs all checks.
test
- Runs the unit tests.
Rules
-----
Pattern: build
Pattern: upload
Pattern: clean
To see all tasks and
more
detail, run with --all.
BUILD SUCCESSFUL
Total
time
: 2.792 secs
|
我们继续,下面要讲怎样打包我们的项目。
我们可以通过使用两个不同的任务来打包项目。
如果我们在命令提示符中执行命令gradle assemble,我们可以看到以下输出:
1
2
3
4
5
6
7
8
9
10
|
> gradle assemble
:compileJava
:processResources
:classes
:jar
:assemble
BUILD SUCCESSFUL
Total
time
: 3.163 secs
|
如果我们在命令提示符中执行命令gradle build,我们可以看到以下输出:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
> gradle build
:compileJava
:processResources
:classes
:jar
:assemble
:compileTestJava
:processTestResources
:testClasses
:
test
:check
:build
BUILD SUCCESSFUL
Total
time
: 3.01 secs
|
这些命令的输出表明了它们的区别:
现在,我们尝试使用以下命令运行我们的程序:
1
|
java -jar first-java-project.jar
|
我们可以看到以下输出:
1
2
|
> java -jar first-java.project.jar
No main manifest attribute,
in
first-java-project.jar
|
问题出在,我们没有在manifest文件中配置Jar文件的主类,让我们继续看看怎样解决这个问题。
Java插件在我们的项目中加入了一个Jar任务,每一个Jar对象都一个manifest属性,这个属性是Manifest的一个实例。
我们可以对生成的Jar文件的主类进行配置,使用Manifest接口的attributes()方法。换句话说,我们可以使用一个包含键值对的map结构指定加入到manifest文件的属性集。
我们能够通过设置Main-Class属性的值,指定我们程序的入口点。在我们对build.gradle文件进行必要的改动后,代码如下:
1
2
3
4
5
6
7
|
apply plugin:
'java'
jar {
manifest {
attributes
'Main-Class'
:
'net.petrikainulainen.gradle.HelloWorld'
}
}
|
(JavaSE教程提供了关于manifest文件的更多信息。)
在我们执行gradle assemble或gradle build命令生成一个新的jar文件之后,我们可以执行以下命令运行jar文件:
1
|
java -jar first-java-project.jar
|
当我们运行程序时,System.out会打印出以下信息:
1
2
|
> java -jar first-java-project.jar
Hello World!
|
这就是我们今天所有的内容,我们看一下我们学到了什么。
我们已经通过Gradle创建了一个简单的Java项目,这篇教程教会了我们四点:
P.S. 这篇教程的示例代码可以在Github找到。