java学习工具----利器

Ant构建

2017-11-04  本文已影响60人  我为峰2014

简介

Apache Ant是一个Java库和命令行工具,其任务是将构建文件中描述的进程作为相互依赖的目标和扩展点。只要使用过Linux系统的读者,应该知道 make这个命令。当编译Linux内核及一些软件源程序时,经常要用这个命令。Make命令其实就 是一个项目管理工具,而Ant所实现功能与此类似,像make,gnumake和nmake这些编译工具都有 一定的缺陷,但是Ant却克服了这些工具的缺陷。

Ant的优点

Ant的开发

Ant的构建文件

当开始一个新的项目时,首先应该编写Ant构建文件。构建文件定义了构建过程,并被团队开发中每个人使用。Ant构建文件默认命名为 build.xml,也可以取其他的名字。只不过在运行的时候把这个命名当作参数传给Ant。构建文件可以放在任何的位置。一般做法是放在项目顶层目录中,这样可以保持项目的简洁和清晰。下面是一个典型的项目层次结构。
( 注:很多项目的ant脚本中的命名基本上都是一致的,比如:编译一般叫build或者compile;打包一般叫jar或war;生成文档一般命名为 javadoc或javadocs;执行全部任务all。在每个任务的中,ANT会根据配置调用一些外部应用并配以相应参数执行。虽然ANT可调用的外部应用种类非常丰富,但其实最常用的就2,3个:比如javac javadoc jar等。)
(1) src存放文件。
(2) class存放编译后的文件。
(3) lib存放第三方JAR包。
(4) dist存放打包,发布以后的代码。

Ant构建文件是XML文件

每个构建文件定义一个唯一的项目(Project元素)。每个项目下可以定义很多目标(target元素),这些目标之间可以有依赖关系。当执行这类目标时,需要执行他们所依赖的目标。每个目标中可以定义多个任务,目标中还定义了所要执行的任务序列。Ant在构建目标时必须调用所定义的任务。任务定义了Ant实际执行的命令。Ant中的任务可以为3类。
(1) 核心任务。核心任务是Ant自带的任务。
(2) 可选任务。可选任务实来自第三方的任务,因此需要一个附加的JAR文件。
(3) 用户自定义的任务。用户自定义的任务实用户自己开发的任务。

Eclipse的集成ANT

打开Eclipse,点击导航栏的"Window"-->"Preferences"-->"Ant"

image.png

创建一个java项目

image.png

在根目录创建一个build.xml文件

image.png

切换默认的Ant版本,Ant的插件管理

image.png

总共需要设置两项:
1、Ant Home Entries(Default),点击Ant Home,选择ant插件路径:
2、设置Global Entries路径,即为jdk中tools.jar路径

Ant自动化构建

使用ant的主要工作就是配置它的xml文件,默认为build.xml文件。

image.png

build.xml的配置参数

1.<project>标签

每个构建文件都有一个对应<project>标签,<project>标签时构建文件的根标签有以下属性:

每个项目对应一个构建文件,但是如果项目比较复杂,包含大量的子项目,每一个子项目都对应有自己的构建文件。

2.<property>标签

有两个特点:

1 、设置 name 和 value 属性值,比如: <property name="srcdir" value="${basedir}/src"/>

2 、 设置 name 和 refid 属性值,比如: <property name="srcpath" refid="dao.compile.classpath"/> ,其中dao.compile.classpath 在别的地方定义。

3 、设置 name 和 location 属性值,比如: <property name="srcdir" location="src"/> ,即将 srcdir 的值设 置为:当前项目根目录的 /src 目录。

4 、设置 file 属性值,比如: <property file="build.properties"/> , 导入 build.properties 属性文件中的属性值

5 、设置 resource 属性值,比如: <propety resource="build.properties"/>, 导入 build.properties 属性文件中的属性值

6 、设置 url 属性值,比如: <property url="http://www.blogjava.net/wiflish/build.properties"/>, 导入http://www.blogjava.net/wiflish/build.properties 属性文件中的属性值。

7 、设置环境变量,比如: <property environment="env"/> ,设置系统的环境变量为前缀 env.
<property name="tomcat.home" value="${env.CATALINA_HOME}"/> 将系统的 tomcat 安装目录设置到 tomcat.home 属性中。

3.<import>标签

4.<target>标签

任务,一个project标签下有一个或多个target标签,代表任务,任务间可以存在依赖关系。有如下属性:

5.<echo>标签

控制台显示,用于打印/输出信息,类似于log4j的info方法

以上四种方式均可以显示相应信息

6.<delete>标签

该标签用于删除文件或文件目录,有如下属性:

<!--clean other dir-->
    <target name="clean_other_dir">
        <echo message="begin clean_other_dir..."/>
        <delete dir="${basedir}/${compress.dir}"/>
        <delete dir="${basedir}/pub"/>
        <echo message="begin clean html module-xx..."/>
        <delete includeemptydirs="true">
            <fileset dir="${basedir}/src/html" >
                <include name="**/module-*/**"/>
            </fileset>
        </delete>
        <echo message="begin clean res/module-xx、component-xx、res-base..."/>
        <delete includeemptydirs="true">
            <fileset dir="${basedir}/res" >
                <include name="module-*/**"/>
                <include name="component-*/**"/>
                <include name="res-base/**"/>
            </fileset>
        </delete>
    </target

7.<mkdir>标签

该标签用于创建一个目录,它有一个属性dir用来指定所创建的目录名,其代码如下:

通过以上代码就创建了一个目录,这个目录已经被前面的property标签所指定。

8.<copy>标签

该标签用于拷贝文件或文件目录,属性如下:

<target name="cp">
    <copy todir="${compress.dir}" overwrite="true">
         <fileset dir="${ob_baseline.dir}">
            <include name="pub/" />
            <include name="res/" />
            <include name="mail_template/" />
         </fileset>
    </copy>
</target>

9.<fileset>标签

文件集标签,通常与任务结合来使用,用于批量cope和delete

10.<exec>执行文件

用来执行系统命令,或者指定环境的命令

打开命名行,并转到c盘执行dir命令

<target name="test">
    <exec executable="cmd.exe">
        <arg line="/c dir"/>
    </exec>
</target>

能够执行系统命令,就相当于可以执行各种环境比如node、gulp、bower等等:

<!--build style-->
<target name="build_style">
    <echo message="begin build_style..."/>
    <exec dir="." executable="gulp" failonerror="true">
        <arg line="scss"/>
    </exec>
</target>

<!--bower cache clean if必须是${]才是判断true,false, 否则只要有设定值即可执行-->
<target name="bower_cache_clean" if="${is_bower_cache_clean}">
    <echo message="begin bower_cache_clean ..."/>
    <exec dir="." executable="bower" failonerror="true">
        <arg line="cache clean" />
    </exec>
</target>

11.<jar>标签

该标签用来生成一个JAR文件,其属性如下

<war>标签

该标签用来生成一个WAR包,其属性如下:

12.<javac标签>

该标签用于编译一个或一组java文件,其属性如下

13.<java>标签

该标签用来执行编译生成的.class文件其属性如下

14.<antcall>标签

AntCall 任务的作用是允许在一个target的执行过程中调用并执行其他的target,AntCall任务必须在target元素内执行,这个任务不能在target元素外执行。

需要从一个target传递参数到被调用的target时,可以使用<param> 类型进行传递,对于param 类型只有两个属性:name和value

<target name="sync_module_teach" if="${is_teach}">
    <antcall target="sync_module_item">
        <param name="html.dir" value="org"/>
    </antcall>
</target>

执行sync_module_item任务,并设置参数html.dir的值为org。
该任务定义如下:

<target name="sync_module_item">
    <echo message="begin sync_module ${html.dir}..."/>
    <copy todir="${basedir}/src/html/${html.dir}" overwrite="true" includeEmptyDirs="true">
        <fileset dir="${basedir}/lib">
            <include name="module-*/**" />
        </fileset>
    </copy>
</target>

或者更为简单的表达:

<target name="deploy">
    <echo message="begin auto deploy......"/>
    <antcall target="clean"/>
    <antcall target="bower_install"/>
    <antcall target="cnpm_install"/>
    <antcall target="sync_module"/>
    <antcall target="build_style"/>
    <antcall target="nej_build" />
    <antcall target="cp"/>
</target>

15.<parallel>标签

并行执行多个子任务

通过failonany控制如果一个失败,则不执行。通过并行执行,来提升性能,降低构建花费的时间

<parallel failonany="true">
    <antcall target="sync_module_corp"/>
    <antcall target="sync_module_main"/>
    <antcall target="sync_module_teach"/>
    <antcall target="sync_module_backend"/>
    <antcall target="sync_module_passport"/>
    <antcall target="sync_module_business"/>
    <antcall target="sync_module_k12_teach"/>
    <antcall target="sync_module_k12_backend"/>

    <antcall target="build_style"/>
</parallel>

**16.<regexp>标签

用于正则的定义的使用,可以与matches结合使用。
比如,定义正则:

<regexp id="regexp_env_test" pattern="^${root_dir}/(${test_dir}|${test_k12_dir})/.+"/>
<regexp id="regexp_env_pre" pattern="^${root_dir}/(${pre_dir}|${pre_k12_dir})/.+"/>

通过pattern指定正则内容,通过id标识。
在需要匹配的时候,使用之:

<condition property="is_test">
    <matches string="${basedir}">
        <regexp refid="regexp_env_test"/>
    </matches>
</condition>

**17.<condition>标签

用来判断,如果包含的内容符合条件,则将property指定的属性设置为true,否则为false。
比如上面的例子中,就是将basedir变量的值和regexp_env_test对应的正则匹配,如果正确,就将is_test设置为true,然后后面的流程再去判断。
与之配合的标签有很多,下面一一介绍:

<condition property="is_test_backend">
    <and>
        <istrue value="${is_test}"/>
        <istrue value="${is_backend}"/>
    </and>
</condition>

只有is_test和is_backend变量的值均为true,is_test_backend的值才为true。

<condition property="scondition">
    <!--如果属性name不存在则返回false-->
    <isset property="name"/>
</condition>
<condition property="scondition">
    <!--如果arg1的值与arg2的值相等返回true,否则为false-->
    <equals arg1="${name}" arg2="this is name"/>
</condition>
<condition property="scondition">
    <!--如果file1所代表的文件与file2所代表的文件相等返回true,否则为false-->
    <filesmatch file1="testfile1.txt" file2="testfile2.txt"/>
</condition>

实例

 <?xml version="1.0" encoding="utf-8"?>

<project name="project name" default="init" basedir=".">

<description>builds, tests, and runs the project. </description>

<!-- ********************************************************

引入资源和定义资源

******************************************************** -->

<!--   引入资源                                  -->

<property file="build.properties"/>

<property environment="env"/>

<!--定义源程序文件夹-->

<property name="src.dir" location="src/java"/>

<property name="test.dir" location="test"/>

<property name="web.dir" location="web"/>

<!--定义目标程序文件夹-->

<property name="build.dir" location="build"/>

<property name="build.classes.dir" location="${build.dir}/classes"/>

<property name="build.test.dir" location="${build.dir}/test"/>

<property name="dist.dir" location="dist"/>

<!--定义其他文件夹-->

<property name="lib.dir" location="lib"/>

<property name="doc.dir" location="doc"/>

<property name="index.dir" location="index"/>

<property name="deploy.dir" location="${env.catalina_home}"/>

<property name="deploy.lib.dir" location="${deploy.dir}/lib"/>

<!--定义其他文件-->

<property name="dist.jar" location="${dist.dir}/web-inf/lib/${project.name}-${project.version}.jar"/>

<property name="deploy.war" location="${deploy.dir}/webapps/${project.name}.war"/>

<!--定义其他属性-->

<available file="${dist.dir}/enduser.agreement" property="final.version"/>

<!-- ********************************************************

设置path

******************************************************** -->

<path id="project.classpath">

<pathelement location="${java.home}/jre/lib/rt.jar"/>

<pathelement location="${build.classes.dir}"/>

<pathelement location="${build.test.dir}"/>

<fileset dir="${deploy.lib.dir}">

<include name="**/*.jar"/>

</fileset>

<fileset dir="${lib.dir}">

<include name="**/*.jar"/>

</fileset>

</path>

<target name="init" description=" 信息 : 显示项目基本信息.">

<tstamp>

<format property="now" pattern="yyyy-mm-dd hh:mm"/>

</tstamp>

<echo> ==================================================


  显示项目基本信息.


  项目名称: ${project.name}

  项目版本: ${project.version}

  作者 : ${author}

   时戳 : ${dstamp}-${tstamp}



  用法:

     ant -buildfile build.xml compile 或

     ant compile 或

     ant 甚至

     ant clean dist

   帮助:

     ant -projecthelp



==================================================</echo>

</target>

<target name="prepare" depends="init" description=" 准备 : 创建各种文件夹.">

<echo> ==================================================



   创建各种文件夹.



================================================== </echo>

<!--  创建源程序文件夹  -->

<mkdir dir="${src.dir}"/>

<mkdir dir="${test.dir}"/>

<mkdir dir="${web.dir}"/>

<mkdir dir="${web.dir}/web-inf"/>

<!--  创建目标程序文件夹  -->

<mkdir dir="${build.dir}"/>

<mkdir dir="${build.classes.dir}"/>

<mkdir dir="${build.test.dir}"/>

<mkdir dir="${dist.dir}"/>

<mkdir dir="${dist.dir}/web-inf"/>

<mkdir dir="${dist.dir}/web-inf/lib"/>

<!--  创建其他文件夹  -->

<mkdir dir="${lib.dir}"/>

<mkdir dir="${doc.dir}"/>

<mkdir dir="${index.dir}"/>

</target>

<target name="javadoc" depends="prepare" description="生成文档: 生成帮助文档.">

<echo> ==================================================



   生成帮助文档.



==================================================</echo>

<javadoc packagenames="*.*" sourcepath="${src.dir}" destdir="${doc.dir}" author="true" version="true" use="true" encoding="utf-8">

<classpath refid="project.classpath"/>

</javadoc>

</target>

<target name="compile" depends="prepare" description=" 编译 : 编译所有源程序." unless="final.version">

<echo> ==================================================



  编译所有源程序.



==================================================</echo>

<javac srcdir="${src.dir}" destdir="${build.classes.dir}" debug="on" deprecation="on" encoding="utf-8">

<compilerarg value="-xlint:unchecked"/>

<classpath refid="project.classpath"/>

</javac>

<javac srcdir="${test.dir}" destdir="${build.test.dir}" encoding="utf-8">

<compilerarg value="-xlint:unchecked"/>

<classpath refid="project.classpath"/>

</javac>

</target>

<target name="test" depends="compile" description=" 测试 : 运行所有测试程序.">

<echo> ==================================================



   运行所有测试程序.



==================================================</echo>

<junit haltonfailure="true">

<classpath refid="project.classpath"/>

<formatter type="brief" usefile="false"/>

<batchtest>

<fileset dir="${build.test.dir}" includes="**/*test.class"/>

</batchtest>

<sysproperty key="doc.dir" value="${doc.dir}"/>

<sysproperty key="index.dir" value="${index.dir}"/>

</junit>

</target>

<target name="dist" depends="compile" description=" 分发 : 生成分发文件.">

<echo> ==================================================



   生成分发文件:

      ${dist.jar}



==================================================</echo>

<!-- 从打包文件排除单元测试 -->

<jar destfile="${dist.jar}" basedir="${build.classes.dir}" includes="**/*.*" excludes="**/*test.class">

<!-- manifest="manifest.mf" > -->

<manifest>

<attribute name="author" value="${author}"/>

</manifest>

</jar>

</target>
<!-- ********************************************************

用于调试

******************************************************** -->

<target name="debug" depends="dist" description=" 调试 "/>

<!-- ********************************************************

用于效验

******************************************************** -->

<target name="verify" depends="dist" description=" 效验 "/>

<target name="run-deploy" depends="dist" description=" 部署 : 把文件部署到指定位置.">

<echo> ==================================================



   把文件部署到指定位置:

      ${deploy.war}



==================================================</echo>

<copy todir="${dist.dir}/web-inf/lib">

<fileset dir="${lib.dir}" includes="*.jar"/>

</copy>

<copy todir="${dist.dir}">

<fileset dir="${web.dir}" includes="**/*.*"/>

</copy>

<jar destfile="${deploy.war}" basedir="${dist.dir}" includes="**/*.*" excludes="**/*test.class">

  <!--  manifest="manifest.mf" >  -->
</jar>
</target>

</project>

模板

以下xml依次定义了init(初始化),compile(编译),test(测试),doc(生成文档),pack(打包)任务,可以作为模板

<?xml version="1.0" encoding="utf-8"?>

<project name="Hello world" default="doc">  

<!-- properies -->  

     <property name="src.dir" value="src" />  

     <property name="report.dir" value="report" />  

     <property name="classes.dir" value="classes" />  

     <property name="lib.dir" value="lib" />  

     <property name="dist.dir" value="dist" />  

<property name="doc.dir" value="doc"/>  

     <!-- 定义classpath -->  

     <path id="master-classpath">  

         <fileset file="${lib.dir}/*.jar" />  

         <pathelement path="${classes.dir}"/>  

     </path>  

     <!-- 初始化任务 -->  

     <target name="init">  

     </target>  

     <!-- 编译 -->  

     <target name="compile" depends="init" description="compile the source files">  

         <mkdir dir="${classes.dir}"/>  

         <javac srcdir="${src.dir}" destdir="${classes.dir}" target="1.4">  

             <classpath refid="master-classpath"/>  

         </javac>  

     </target>  

     <!-- 测试 -->  

     <target name="test" depends="compile" description="run junit test">  

         <mkdir dir="${report.dir}"/>  

         <junit printsummary="on"  

                 haltonfailure="false"  

                 failureproperty="tests.failed"  

                 showoutput="true">  

             <classpath refid="master-classpath" />  

             <formatter type="plain"/>  

             <batchtest todir="${report.dir}">  

                 <fileset dir="${classes.dir}">  

                     <include name="**/*Test.*"/>  

                 </fileset>  

             </batchtest>  

         </junit>  

         <fail if="tests.failed">  

         ***********************************************************   

         ****   One or more tests failed!   Check the output ...   ****   

         ***********************************************************   

         </fail>  

     </target>  

     <!-- 打包成jar -->  

     <target name="pack" depends="test" description="make .jar file">  

      <mkdir dir="${dist.dir}" />  

         <jar destfile="${dist.dir}/hello.jar" basedir="${classes.dir}">  

             <exclude name="**/*Test.*" />  

             <exclude name="**/Test*.*" />  

         </jar>  

     </target>  

     <!-- 输出api文档 -->  

     <target name="doc" depends="pack" description="create api doc">  

      <mkdir dir="${doc.dir}" />  

      <javadoc destdir="${doc.dir}"  

             author="true"  

             version="true"  

             use="true"  

             windowtitle="Test API">  

             <packageset dir="${src.dir}" defaultexcludes="yes">  

                 <include name="example/**" />  

             </packageset>  

             <doctitle><![CDATA[<h1>Hello, test</h1>]]></doctitle>  

             <bottom><![CDATA[<i>All Rights Reserved.</i>]]></bottom>  

             <tag name="todo" scope="all" description="To do:" />  

         </javadoc>  

     </target>  

</project>  
上一篇下一篇

猜你喜欢

热点阅读