Maven打包Jar

现状

该项目使用了Maven,并且引入了Spring,包含代码、配置文件、Jar包,使用的是IDEA来作为开发工具,项目的产出物是要打包成一个可运行的Jar包。通过IDEA的打包工具也可以打包成功,只不过需要自己向MANIFEST.MF里面写入Class-Path(引入的Jar包多的时候就比较烦了,或许有简单的方法是我不知道的吧)。鉴于项目使用了Maven,且Maven里面拥有众多优秀的插件,最终决定使用Maven来作为项目的打包工具。

目标

  • 自己的代码编译后作为一个主要的启动Jar
  • 自己的源代码也打包成为一个Jar
  • 引用的Jar包放入到lib目录下
  • 需要的配置文件放在conf目录下
  • 将我们的启动Jar、lib目录、conf目录打包成一个文件

准备

  • 工具
    • Maven(必需的)
    • IDEA(非必需,看个人习惯)
  • Maven插件(核心的)
    • maven-compiler-plugin
    • maven-jar-plugin
    • maven-dependency-plugin
    • maven-resources-plugin
    • maven-source-plugin
    • maven-assembly-plugin

POM详解

基本属性

<project>
	[...]
  	<properties>
      	<!--项目使用的JDK版本-->
        <jdk.version>1.8</jdk.version>
      	<!--项目使用的Spring版本-->
        <spring.version>4.3.10.RELEASE</spring.version>
      	<!--项目文件编码-->
        <sourceEncoding>UTF-8</sourceEncoding>
    </properties>
  	[...]
  	<dependencies>
      <!--项目的依赖包-->
      [...]
  	</dependencies>
</project>

插件配置

<project>
  [...]
  <build>
    <plugins>
      <!--我们要使用的插件就在这里-->
      [...]
    </plugins>
  </build>
  [...]
</project>

maven-compiler-plugin

这个插件是maven在编译项目的时候需要的信息,主要是为其指明JDK版本信息以及编码方式。一般的maven项目初始化后的默认JDK版本是1.5的,如果不注意就会出现莫名其妙的错误。

<plugin>
  <groupId>org.apache.maven.plugins</groupId>
  <artifactId>maven-compiler-plugin</artifactId>
  <version>3.6.2</version>
  <configuration>
    <source>${jdk.version}</source>
    <target>${jdk.version}</target>
    <encoding>${project.build.sourceEncoding}</encoding>
  </configuration>
</plugin>

maven-jar-plugin

这个插件就是用在打包时,指明必要的信息的,它还有一些其他的定制化的设置可以参考 官方文档

<plugin>
  <groupId>org.apache.maven.plugins</groupId>
  <artifactId>maven-jar-plugin</artifactId>
  <configuration>
    <archive>
      <!-- 清单文件 -->
      <manifest>
        <!-- 指明我们的项目启动入口,即包含main方法的那个类 -->
        <mainClass>cn.xxx.Abcd</mainClass>
        <!-- 需要增加类的加载路径 -->
        <addClasspath>true</addClasspath>
        <!-- 
			类的加载路径,这里的路径是以打包后的jar文件为基准的
			也就是说这个lib就在打包后的jar所在的那个文件夹下
 		-->
        <classpathPrefix>lib/</classpathPrefix>
      </manifest>
      <manifestEntries>
        <!--
			这里的conf会添加到maven打包时产生的MANIFEST.MF的Class-Path节点中
			通常的,我们在Spring中的配置文件都是通过classpath来指明位置的,所以这里的说明可以让我们把配置文件拿出来单独放在一起
		-->
        <Class-Path>conf/</Class-Path>
      </manifestEntries>
    </archive>
  </configuration>
</plugin>

maven-dependency-plugin

这个插件的作用就是把一些依赖包拷贝或者解压到指定的位置去。我们这里仅用了其copy的目标,它的其他目标请参考 官方文档

<!-- 拷贝依赖的jar包到lib目录 -->
<plugin>
  <groupId>org.apache.maven.plugins</groupId>
  <artifactId>maven-dependency-plugin</artifactId>
  <executions>
    <execution>
      <id>copy</id>
      <!--绑定到maven的生命周期的package阶段-->
      <phase>package</phase>
      <goals>
        <!--指明我们的目标:拷贝依赖-->
        <goal>copy-dependencies</goal>
      </goals>
      <configuration>
        <!--
			指明我们拷贝的依赖要放到哪个文件夹下面
			这里指的是项目的: /项目根目录/target/lib 目录下
		-->
        <outputDirectory>
          ${project.build.directory}/lib
        </outputDirectory>
      </configuration>
    </execution>
  </executions>
</plugin>

maven-resources-plugin

这个主要用来指明资源文件的编码方式了,但是这么插件还有许多其他的功能,具体详情请参考 官方文档

<plugin>
  <groupId>org.apache.maven.plugins</groupId>
  <artifactId>maven-resources-plugin</artifactId>
  <version>2.5</version>
  <configuration>
    <encoding>${project.build.sourceEncoding}</encoding>
  </configuration>
</plugin>

maven-source-plugin

这个插件用来把我们的源代码也打包成一个Jar文件,这里我们使用其一部分属性,其余属性的使用请参考 官方文档

<plugin>
  <artifactId>maven-source-plugin</artifactId>
  <version>2.1</version>
  <configuration>
    <!--这里你可以添加一个outputDirectory节点来指明这个文件的输出目录,fileName来指明其输出的名字-->
    <attach>true</attach>
  </configuration>
  <executions>
    <execution>
      <phase>compile</phase>
      <goals>
        <goal>jar</goal>
      </goals>
    </execution>
  </executions>
</plugin>

maven-assembly-plugin

好了,怎么打包,前面都做完了,那么这个插件就是来给他们分分类,然后做成一个jar、zip或者tar.gz等等,把我们需要整合到一起的东西给打包到一起。当然,你还可以通过看 官方文档 来进一步了解它。

其实到这一步之前,我们的jar包已经打包完成了,我们的依赖包也都已经拷贝到target/lib下了,同样的,我们的配置文件也已经放到了target/conf下面了,而且我们也已经可以通过java -jar Abcd.jar来运行了。所以,这个插件就是用来把这些东西都放在一起,来做最后一步——整合。

<plugin>
  <artifactId>maven-assembly-plugin</artifactId>
  <version>3.1.0</version>
  <configuration>
    <appendAssemblyId>false</appendAssemblyId>
    <descriptors>
      <!--这个插件需要一个说明器,这里我将它放在src下的assembly目录下-->
      <descriptor>src/assembly/assembly.xml</descriptor>
    </descriptors>
  </configuration>
  <executions>
    <execution>
      <!--这个插件需要绑定到package阶段执行-->
      <phase>package</phase>
      <goals>
        <!--这里是指只执行一次-->
        <goal>single</goal>
      </goals>
    </execution>
  </executions>
</plugin>

然后我们在assembly.xml如下所示

<assembly xmlns="http://maven.apache.org/ASSEMBLY/2.0.0"
          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
          xsi:schemaLocation="http://maven.apache.org/ASSEMBLY/2.0.0 http://maven.apache.org/xsd/assembly-2.0.0.xsd">
    <id>distribution</id>
    <formats>
      	<!--最后要输出的文件格式
		有:zip、tar、tar.gz (or tgz)、tar.bz2 (or tbz2)、tar.snappy、tar.xz (or txz)、jar、dir、war可以选择
		-->
        <format>zip</format>
        <format>tar.gz</format>
    </formats>
  	<!--下面就可以看作是分类的过程了-->
    <fileSets>
        <fileSet>
          	<!--这里指明我们的源文件所在处-->
            <directory>${basedir}</directory>
          	<!--这里指明我们的输出目录-->
            <outputDirectory>\</outputDirectory>
            <!--这里来说明哪些需要包含进来,当然也可以用excludes来说明不包含哪些文件-->
          	<includes>
                <include>README*</include>
                <include>LICENSE*</include>
                <include>NOTICE*</include>
            </includes>
        </fileSet>
        <fileSet>
            <directory>${project.basedir}\src\script</directory>
            <outputDirectory>\</outputDirectory>
            <includes>
                <include>startup.*</include>
                <include>shutdown.*</include>
            </includes>
        </fileSet>
        <fileSet>
            <directory>${project.build.directory}\lib</directory>
            <outputDirectory>\lib</outputDirectory>
            <includes>
                <include>*.jar</include>
            </includes>
        </fileSet>
        <fileSet>
            <directory>${project.build.directory}</directory>
            <outputDirectory>\</outputDirectory>
            <includes>
                <include>${project.build.finalName}.jar</include>
            </includes>
        </fileSet>
        <fileSet>
            <directory>${project.build.directory}\conf</directory>
            <outputDirectory>\conf</outputDirectory>
            <includes>
                <include>*.properties</include>
            </includes>
        </fileSet>
    </fileSets>
    <dependencySets>
        <dependencySet>
          	<!--这里用来指明当打包时我们的依赖文件(*.jar)不要解压,要原封不动的打包进去-->
            <unpack>false</unpack>
            <outputDirectory>\lib</outputDirectory>
            <useProjectArtifact>false</useProjectArtifact>
            <!-- 将scope为runtime的依赖包打包到lib目录下。 -->
            <scope>runtime</scope>
        </dependencySet>
    </dependencySets>
</assembly>

资源说明

我们还需要说明哪些是我们的资源文件,然后这些会打包进我们的Jar文件里去,而打包进去的资源文件自然地可以以类路径访问了。

<resources>
  <!-- 控制资源文件的拷贝 -->
  <resource>
    <!--这里来指明那个文件夹是我们的资源文件夹-->
    <directory>src/main/resources</directory>
    <includes>
      <!--来指明需要包含的文件,这里用通配符来表示了,即指在resources下的任意子目录(包括其本身)下的xml和tld文件-->
      <include>**/*.xml</include>
      <include>**/*.tld</include>
    </includes>
    <!--指明这里不需要过滤,注意过滤的文件是不会被打包进去的,就像下面的*.properties文件都不会被打包-->
    <filtering>false</filtering>
  </resource>
  <resource>
    <directory>src/main/resources</directory>
    <includes>
      <include>**/*.properties</include>
    </includes>
    <filtering>true</filtering>
    <!--上面的filter指明这类文件需要过滤出来,那么接下来这个节点说明这类文件还需要输出到/conf目录下-->
    <targetPath>${project.build.directory}/conf</targetPath>
  </resource>
</resources>

最后

经过以上的步骤之后,我们的项目就算打包完毕了,整个项目目录就是下面这个样子的

\root
 |``\lib
 |	 ``\*.jar
 |``\conf
 |	 ``\*.properties
 |``\Abcd.jar

还有一个跨平台的脚本插件appassembler-maven-plugin,可是我觉得shell这种东西,自己写一写会更好,所以这里仅提供其名字,不提供其用法了。

posted @ 2017-10-11 19:48  风如杨  阅读(835)  评论(0)    收藏  举报