介绍使用 Ant 打包可运行的 Jar 包。

打包 jar 包最大的问题在于如何加入第三方 jar 包使得 jar 文件可以直接运行,以下用实例进行说明。

程序结构:

关键代码:

package com.alfred.main;

import com.alfred.bean.User;
import com.alfred.util.ProjConfig;
import com.thoughtworks.xstream.XStream; public class Main {
public static void main(String[] args) {
User user = new User();
user.setUsername("alfred");
user.setAge(15);
XStream xstream = new XStream();
String xml = xstream.toXML(user);
System.out.println("in Main");
System.out.println("prop:"+ProjConfig.MY_PROPERTY);
System.out.println("xml:"+xml);
}
}

Main.java

<?xml version="1.0" encoding="UTF-8"?>
<project basedir="." default="usage" name="antproj">
<!-- 引入配置文件 -->
<property file="build.properties" />
<!-- 配置引用属性 -->
<property name="src.dir" value="src" />
<property name="config.dir" value="config" />
<property name="build.dir" value="build" />
<property name="lib.dir" value="lib" />
<property name="name" value="antproj" /> <path id="master-classpath">
<fileset dir="${lib.dir}">
<include name="*.jar" />
</fileset>
<pathelement path="${build.dir}" />
</path> <target name="usage">
<!-- 输出信息到控制台中 -->
<echo message="" />
<echo message="${name} build file" />
<echo message="-----------------------------------" />
<echo message="" />
<echo message="Available targets are:" />
<echo message="" />
<echo message="build --> Build the application" />
<echo message="" />
</target> <!-- 编译生成class文件 -->
<target name="build"
description="Compile main source tree java files"> <!-- 删除编译存放的classes文件 -->
<delete dir="${build.dir}/classes"></delete>
<!-- 新建编译存放的classes文件 -->
<mkdir dir="${build.dir}/classes" />
<copy todir="${build.dir}/classes">
<!-- 将项目中除会编译生成class文件的java文件之外其他类型的文件拷贝到对应的目录下,指定文件名后缀 -->
<fileset dir="${src.dir}">
<include name="**/*.properties" />
<include name="**/*.xml" />
</fileset>
<fileset dir="${config.dir}">
<include name="**/*.properties" />
</fileset>
</copy>
<!-- java编译 -->
<javac destdir="${build.dir}/classes" source="1.5" target="1.5"
debug="true" deprecation="false" optimize="false" failonerror="true"
encoding="utf-8">
<src path="${src.dir}" />
<!-- 编译所需要的jar包路径 -->
<classpath refid="master-classpath" />
</javac>
</target> <target name="jar" depends="build"
description="Compress the java class files to jar"> <mkdir dir="${build.dir}/jar" />
<delete file="${build.dir}/jar/${name}.jar" />
<buildnumber file="${build.dir}/buildnum.txt" />
<!-- 指定时间戳 可以调用TODAY -->
<tstamp>
<format property="TODAY" pattern="yyyy-MM-dd HH:mm:ss" />
</tstamp> <!-- 生成清单文件 -->
<manifest file="${build.dir}/MANIFEST.MF">
<attribute name="Built-By" value="${user.name}" />
<attribute name="Implementation-Version"
value="${version.num}.${build.number}" />
<attribute name="Built-Date" value="${TODAY}" />
<attribute name="Main-Class" value="com.alfred.main.Main" />
</manifest> <!-- 打包jar -->
<!-- includes 包含哪些后缀文件 空格分隔 -->
<jar destfile="${build.dir}/jar/${name}.jar"
basedir="${build.dir}/classes"
includes="**/*.class **/*.properties **/*.xml"
manifest="${build.dir}/MANIFEST.MF">
<zipfileset src="${lib.dir}/xstream-1.4.9.jar"></zipfileset>
<zipfileset src="${lib.dir}/xpp3-1.1.3.3.jar"></zipfileset>
</jar>
</target> <target name="cleanup">
<!-- 删除编译文件夹下的所有文件(不包括编译文件夹下的文件夹)
<delete>
<fileset dir="${build.dir}/classes" includes="**/*.*" />
</delete>
-->
<!-- 删除编译文件夹 -->
<delete dir="${build.dir}/classes" />
</target>
</project>

build.xml

将第三方 jar 包加入的关键代码是:

<zipfileset src="${lib.dir}/xstream-1.4.9.jar"></zipfileset>
<zipfileset src="${lib.dir}/xpp3-1.1.3.3.jar"></zipfileset>

这种情况下生成的 jar 包内部结构如下:

可以看到第三方 jar 包实际上都被拆开打包进我们的 jar 包了。如果使用的第三方 jar 包太多的话,会变得非常混乱。这时可以通过单独将程序打包为 jar 包,通过引用的方法调用外部第三方 jar 包的方式运行。

如果使用如下的写法:

<zipfileset dir="${lib.dir}">
<include name="*.jar"/>
</zipfileset>

那么打包生成的 jar 包将无法正常调用第三方 jar 包,jar 包结构如下:

可以看到第三方 jar 包都是完整的。

由以上示例可知,无论是将第三方 jar 包拆分打包(结构混乱)还是保持第三方 jar 包完整打包(无法正常运行),都存在一定的问题,以下介绍一种通过引用的方式调用第三方 jar 包的方式。

在 build.xml 中新建一个任务

<target name="jar2" depends="build" description="Compress the java class files to jar">
<mkdir dir="${build.dir}/jar" />
<delete file="${build.dir}/jar/${name}.jar" />
<buildnumber file="${build.dir}/buildnum.txt" />
<!-- 指定时间戳 可以调用TODAY -->
<tstamp>
<format property="TODAY" pattern="yyyy-MM-dd HH:mm:ss" />
</tstamp> <pathconvert property="quote.classpath" pathsep=" ">
<mapper>
<chainedmapper>
<!-- jar包文件只留文件名,去掉目录信息 -->
<flattenmapper />
<!-- add lib/ prefix -->
<globmapper from="*" to="lib/*" />
</chainedmapper>
</mapper>
<path refid="master-classpath" />
</pathconvert> <!-- 生成清单文件 -->
<manifest file="${build.dir}/MANIFEST.MF">
<attribute name="Built-By" value="${user.name}" />
<attribute name="Implementation-Version" value="${version.num}.${build.number}" />
<attribute name="Built-Date" value="${TODAY}" />
<attribute name="Main-Class" value="com.alfred.main.Main" />
<attribute name="Class-Path" value="${quote.classpath}" />
</manifest> <!-- 打包jar -->
<!-- includes 包含哪些后缀文件 空格分隔 -->
<jar destfile="${build.dir}/jar/${name}.jar"
basedir="${build.dir}/classes"
includes="**/*.class **/*.properties **/*.xml"
manifest="${build.dir}/MANIFEST.MF">
</jar>
</target>

将关联的第三方 jar 包通过路径转换加上路径前缀(也就是之后你 jar 包引用第三方 jar 包的路径),生成新的路径信息 quote.classpath,将路径信息配置进 Class-Path 属性,通过这种方式生成的 jar 包中 MANIFEST.MF 文件内容中 Class-Path 如下:

Class-Path: lib/xpp3-1.1.3.3.jar lib/xstream-1.4.9.jar

之后我们将第三方 jar 包放置于 lib 目录中,将打包后的 antproj.jar 放置于和 lib 同级目录,运行 jar 正常。

另外,要打包可运行的 jar 包,加入第三方 jar 包的话,也可以通过一些 IDE 的插件,例如:fatjar。

Ant打包可运行的Jar包(加入第三方jar包)的更多相关文章

  1. Java项目导出为jar包+导出第三方jar包+使用命令行调用+传参

    Java项目导出为jar包+导出第三方jar包+使用命令行调用+传参 一.打包 情况1:不需要向程序传参数,并且程序没有使用第三方jar包 Eclipse上导出jar: 然后选择一个java文件作为入 ...

  2. eclipse打包jar时包含第三方jar包的相关问题

    我用的是mars4.5版本的eclipse 需求:要把写好的工程打成jar包,并能直接运行.工程用了若干个第三方jar. 在打包的时候,eclipse提供的打包方法不能引用第三方jar包,导致了出现C ...

  3. myeclipse8.5打包jar并引入第三方jar包

    用myeclipse自带的export工具,无法引入被引用的第三方的jar包,有两种方式: (1)直接export出jar包,解压jar包(第三方的jar包太多,太麻烦) 在创建好的JAR文件里找到M ...

  4. 用eclipse打包可执行的jar(含第三方jar包)

    在eclipse中的解决方式如下: 在工程目录下(与src同层)建立lib目录,将第三方Jar包放到这个目录里(copy,paste即可)[如果直接引用本地的jar,一旦换电脑就呵呵了...] 右击工 ...

  5. 将Maven项目打包成可执行jar文件(引用第三方jar)

    方法一. mvn assembly 或 mvn package (一个jar包) 把依赖包和自己项目的文件打包如同一个jar包(这种方式对spring的项目不支持) <build>     ...

  6. eclipse打jar包解决第三方依赖包

    在项目根目录下手动MANIFEST.MF(eclipse无法自动生成) MANIFEST.MF Manifest-Version: 1.0 Class-Path: lib/kafka-clients- ...

  7. 用ant打包可运行的jar文件 (将第三方jar包放进你自己的jar包)

    http://blog.csdn.net/caiqcong/article/details/7618582 <span style="font-family:SimSun;font-s ...

  8. Maven打包jar-打包jar时引入第三方jar

  9. 将Maven项目打包成可执行 jar文件(引用第三方jar)

    使用maven assembly插件完成打包 修改pom: <build> <pluginManagement> <plugins> <!--设置jdk版本, ...

随机推荐

  1. Python库源码学习1:Flask之app.run

    先列出app.run()实现的功能,我们以debug=True的情况下进行分析. 1. web服务器,处理http请求 2. 当代码修改后,重启服务器 那么app.run()是如何实现这两个功能的呢? ...

  2. 基于w2v词向量聚类出现的问题(有待解决)

    1.训练词向量代码如下:#训练词语为向量表示def w2v_train(self): ques = self.cu.execute('select question from activity')#将 ...

  3. 第 7 章 Data 类型

    目录 第 7 章 Data 类型 一.创建方式 二.转时间戳 其他 第 7 章 Data 类型 @(es5) 参考了: 阮一峰javascript的标准.<javascript高级教程> ...

  4. oracle查看哪些表被锁

    select b.owner,b.object_name,a.session_id,a.locked_mode from v$locked_object a,dba_objects b where a ...

  5. IQ调制原理

    现代通信中,IQ调制基本上属于是标准配置,因为利用IQ调制可以做出所有的调制方式. 但是IQ调制到底是怎么工作的,为什么需要星座映射,成型滤波又是用来干嘛的.这个呢,讲通信原理的时候倒是都会泛泛的提到 ...

  6. python-面向对象-02_类和对象

    类和对象 目标 类和对象的概念 类和对象的关系 类的设计 01. 类和对象的概念 类 和 对象 是 面向对象编程的 两个 核心概念 1.1 类 类 是对一群具有 相同 特征 或者 行为 的事物的一个统 ...

  7. 简单好用的计算器:bc

    1.简单好用的计算器:bc (1)参数: (2)实例: 执行浮点运算和一些高级函数 设定小数精度(数值范围) 进制转换 执行结果为:11000000,这是用bc将十进制转换成二进制 计算平方和平方根:

  8. python 内置方法join 给字符串加分隔符

    #!/usr/bin/python3 # -*- coding: utf-8 -*- test = "今天吃了吗" test = "_".join(test) ...

  9. 对于jquery实现原理的浅谈

    关键词:prototype(原型).它能让javascript的方法(也可看成:类)能够动态地追加方法(猜测:目的是为了代码实现引入“类的思想”) 废话少说,代码见真义. <html> & ...

  10. nodejs+mysql入门实例(改)

    //连接数据库 var mysql = require('mysql'); var connection = mysql.createConnection({ host: 'bdm253137448. ...