孤楓舞影

          Victory won't come to me unless I go to it.

            BlogJava :: 首頁(yè) :: 聯(lián)系 :: 聚合  :: 管理
            5 Posts :: 0 Stories :: 2 Comments :: 0 Trackbacks

          Ant教程(一)

          本文以最新發(fā)布的Ant 1.7.1為例,介紹這款優(yōu)秀的Build工具的安裝配置、基本應(yīng)用和一些高級(jí)話題。最新的Ant下載地址是 http://jakarta.apache.org/ant/

          Ant是一種基于Java的Build工具。理論上來(lái)說(shuō),它有些類似于C中的make,但比make優(yōu)越。現(xiàn)在存在的大多數(shù)Build工具,如make、gnumake、nmake、jam等都存在這樣或那樣的不足,比如依賴于特定的平臺(tái)、配置文件過(guò)于復(fù)雜或者對(duì)格式無(wú)法檢查而容易出錯(cuò)等。與這些工具相比較,Ant的兩個(gè)特性決定了它是一款優(yōu)秀的Build工具:

          1. 基于Java的實(shí)現(xiàn)。具有良好的跨平臺(tái)性,同時(shí)可以通過(guò)增加新的Java類來(lái)擴(kuò)展Ant的功能,而無(wú)需去了解不同平臺(tái)上不同的腳本語(yǔ)言。

          2.基于XML的配置文件。Ant以XML樹(shù)來(lái)描述Target/Task的關(guān)系,文件結(jié)構(gòu)清晰、易讀易寫(xiě),并且利用XML對(duì)格式的控制來(lái)避免由于配置文件的錯(cuò)誤造成的Build操作失敗。

          安裝與配置

          Ant的安裝非常簡(jiǎn)單,把從網(wǎng)上下載的jakarta-ant-1.5.1-bin.zip解開(kāi)到一個(gè)目錄下即可(以下假定安裝在目錄D:\jakarta-ant-1.5.1)。接下來(lái)需要進(jìn)行環(huán)境變量配置:

          SET ANT_HOME=D:\jakarta-ant-1.5.1 //注意是Ant的安裝目錄,不是bin子目錄
          SET PATH=%PATH%;%ANT_HOME%\bin;


          在配置環(huán)境變量之前,請(qǐng)確認(rèn)已經(jīng)正確設(shè)置了JAVA_HOME系統(tǒng)變量。輸入ant命令,看到如下輸出說(shuō)明已成功安裝了Ant工具:

          Buildfile: build.xml does not exist!
          Build failed


          提示信息表明在當(dāng)前目錄不存在build.xml配置文件,但這本身已經(jīng)說(shuō)明Ant成功運(yùn)行了。

          快速入門(mén)

          下面用一個(gè)最簡(jiǎn)單也是最經(jīng)典的例子-HelloWorld來(lái)感受一下Ant吧。

          //HelloWorld.java
          package com.sharetop.antdemo;
          public class HelloWorld {
          public static void main( String args[] ) {
          System.out.println("Hello world. ");
          }
          }


          要讓Ant編譯這個(gè)文件,首先需要編寫(xiě)一個(gè)Build配置文件。在一般情況下,這個(gè)文件被命名為build.xml。

          <?xml version="1.0" encoding="UTF-8" ?>
          <project name="HelloWorld" default="run" basedir="." >
          <property name="src" value="src"/>
          <property name="dest" value="classes"/>
          <property name="hello_jar" value="hello.jar" />
          <target name="init">
          <mkdir dir="${dest}"/>
          </target>
          <target name="compile" depends="init">
          <javac srcdir="${src}" destdir="${dest}"/>
          </target>
          <target name="build" depends="compile">
          <jar jarfile="${hello_jar}" basedir="${dest}"/>
          </target>
          <target name="run" depends="build">
          <java classname="com.sharetop.antdemo.HelloWorld" classpath="${hello_jar}"/>
          </target>
          </project>


          來(lái)看一下這個(gè)文件的內(nèi)容,它描述了以下信息:工程的名字為HelloWorld,工程有四個(gè)target,分別是init、compil、build和run,缺省是run。compile只有一個(gè)任務(wù)javac,源文件位于src目錄下,輸出的類文件要放在classes目錄下。build的任務(wù)是jar,生成的jar文件為hello.jar,它打包時(shí)以classes為根目錄。而run則是執(zhí)行這個(gè)HelloWorld類,用hello.jar作為classpath。這四個(gè)target之間有一個(gè)依賴關(guān)系,這種關(guān)系用depends來(lái)指定。即如果Target A依賴于Target B,那么在執(zhí)行Target A之前會(huì)首先執(zhí)行Target B。所以從下面運(yùn)行缺省Target(run)的輸出看,這四個(gè)Target的執(zhí)行順序是:init→compile→build→run。文件目錄結(jié)構(gòu)如圖1所示。HelloWorld.java文件在src\com\sharetop\antdemo子目錄下。


          圖1 ant_demo應(yīng)用的目錄結(jié)構(gòu)


          在命令行輸入命令:ant,然后運(yùn)行,可以看到如下輸出:

          如果配置文件名不是build.xml,比如是build_front.xml,那么,可以使用-buildfile命令參數(shù)指定:

          G:\myDoc\ant_demo>ant -buildfile build_front.xml


          也可以單獨(dú)執(zhí)行指定的某個(gè)target,比如,只編譯不打包執(zhí)行,可以使用下面輸入命令即可:

          G:\myDoc\ant_demo>ant compile

          在相應(yīng)的目錄下會(huì)找到編譯出的HelloWorld.class文件。

          再看看上面的build.xml配置文件,文件開(kāi)頭定義了3個(gè)屬性,分別指定了源文件輸出路徑、類文件輸出路徑和生成的Jar文件名,后面對(duì)這些路徑的引用都通過(guò)一個(gè)${property name}來(lái)引用。所以,要注意這樣一個(gè)原則“目錄的定義與目錄的引用應(yīng)該分開(kāi)”。
           

          Ant教程(二)

          基本應(yīng)用

          建立工程的目錄

          一般要根據(jù)工程的實(shí)際情況來(lái)建立工程的目錄結(jié)構(gòu)。但是,有一些比較通用的組織形式可供參考,比如所有的jakarta項(xiàng)目都使用類似的目錄結(jié)構(gòu)。下面讓我們來(lái)看一下這種目錄結(jié)構(gòu)的特點(diǎn)。

          表1

          目錄 文件
          bin 公共的二進(jìn)制文件,以及運(yùn)行腳本
          build 臨時(shí)創(chuàng)建的文件,如類文件等
          dist 目標(biāo)輸出文件,如生成Jar文件等。
          doc/javadocs 文檔。
          lib 需要導(dǎo)出的Java包
          src 源文件

          對(duì)于一個(gè)簡(jiǎn)單的工程,一般包括表1的幾個(gè)目錄。其中bin、lib、doc和src目錄需要在CVS的控制之下。當(dāng)然在這樣的目錄結(jié)構(gòu)上,也可以做一些調(diào)整,例如,可以建立一個(gè)extra目錄來(lái)放置需要發(fā)布的Jar文件、Inf文件及圖像文件等。同樣,如果開(kāi)發(fā)Web應(yīng)用可以建立一個(gè)Web目錄放置JSP、HTML等文件。

          如果我們開(kāi)發(fā)的是一個(gè)比較復(fù)雜的項(xiàng)目,包括多個(gè)子項(xiàng)目,并且各個(gè)子項(xiàng)目是由不同的開(kāi)發(fā)人員來(lái)完成的,那么要如何來(lái)設(shè)計(jì)它的目錄結(jié)構(gòu)?首先有一點(diǎn)是需要確定的,不同的子項(xiàng)目應(yīng)該擁有不同的Build文件,并且整個(gè)項(xiàng)目也應(yīng)該有一個(gè)總的Build文件。可以通過(guò)Ant任務(wù)或是AntCall任務(wù)調(diào)用子項(xiàng)目的Build文件,如下例:

          <target name="core" depends="init">
          <ant dir="components" target="core"/>
          <ant dir="waf/src" target="core"/>
          <ant dir="apps" target="core"/>
          </target>


          在各個(gè)子項(xiàng)目的耦合不是非常緊密的情況下,各個(gè)子項(xiàng)目應(yīng)該有各自獨(dú)立的目錄結(jié)構(gòu),也就是說(shuō)它們可以有自己的src、doc、build、dist等目錄及自己的build.xml文件,但是可以共享lib和bin目錄。而對(duì)于那些耦合緊密的子項(xiàng)目,則推薦使用同一個(gè)src目錄,但是不同的子項(xiàng)目有不同的子目錄,各個(gè)子項(xiàng)目的build.xml文件可以放在根目錄下,也可以移到各個(gè)子項(xiàng)目的目錄下。

          編寫(xiě)B(tài)uild文件

          要用好Ant工具,關(guān)鍵是要編寫(xiě)一個(gè)build.xml文件。要編寫(xiě)出一個(gè)結(jié)構(gòu)良好、靈活可擴(kuò)展的Build文件,有兩個(gè)問(wèn)題要考慮,一是了解Build文件的基本結(jié)構(gòu),二是了解Ant定義的大量任務(wù)。

          Ant的Build文件是一個(gè)標(biāo)準(zhǔn)的XML文件,它包含一個(gè)根節(jié)點(diǎn)Project,每個(gè)Project定義了至少一個(gè)或多個(gè)Target,每個(gè)Target又是一系列Task的集合。它們之間的關(guān)系如圖2所示。


          圖2 build.xml文件的結(jié)構(gòu)


          每個(gè)Task是一段可被執(zhí)行的代碼,比如,前例中的javac、jar就是兩個(gè)最常用的Task。Ant定義了大量的核心Task,我們要考慮的第二個(gè)問(wèn)題正是如何去掌握這大量的Task。其實(shí)唯一的方法就是邊學(xué)習(xí)邊實(shí)踐,這方面最好的參考就是官方的Ant使用手冊(cè)。

          外部文件的使用

          使用外部的Property文件可以保存一些預(yù)設(shè)置的公共屬性變量。這些屬性可以在多個(gè)不同的Build文件中使用。

          可以將一個(gè)外部的XML文件導(dǎo)入Build文件中,這樣多個(gè)項(xiàng)目的開(kāi)發(fā)者可以通過(guò)引用來(lái)共享一些代碼,同樣,這也有助于Build文件的重用,示例代碼如下所示:

          <!DOCTYPE project [
          <!ENTITY share-variable SYSTEM "file:../share-variable.xml">
          <!ENTITY build-share SYSTEM "file:../build-share.xml">
          ]>
          <project name="main" default="complie" basedir=".">
          &share-variable;
          &build-share;
          ... ...


          在J2EE項(xiàng)目中的應(yīng)用

          只要掌握了Ant的使用方法,在J2EE項(xiàng)目中的應(yīng)用與在其它項(xiàng)目中的應(yīng)用并沒(méi)有太大的不同,但是仍有幾點(diǎn)是需要注意的。

          一是要清楚War和Jar文件的目錄結(jié)構(gòu),主要是War的配置文件web.xml文件的位置和EJB的配置文件(ejb-jar.xml和weblogic-ejb-jar.xml等)的位置,在調(diào)用Jar任務(wù)打包文件時(shí)一定要記得把它們也包含進(jìn)來(lái)。一般在編譯之前就要注意把這些需打包的文件拷入相應(yīng)目錄下。二是在J2EE項(xiàng)目中可能會(huì)涉及到一些特殊的任務(wù),比如在Weblogic中會(huì)調(diào)用ejbc預(yù)編譯EJB的代碼存根,或者需要在Ant中同時(shí)發(fā)布Jar到相應(yīng)的服務(wù)器中等。可以用兩種途徑實(shí)現(xiàn)這些任務(wù),一是擴(kuò)展Ant任務(wù)實(shí)現(xiàn)這些任務(wù),二是直接用Java任務(wù)來(lái)執(zhí)行這些命令。下面是打包、發(fā)布一個(gè)EJB的build.xml配置文件片斷,代碼如下:

          <target name="deploy_HelloEJB" depends="compile">
          <delete dir="${temp}/ejb_make"/> <!-- 首先刪除臨時(shí)目錄 -->
          <delete file="${temp}/helloEJB.jar"/>
          <!-- 刪除WebLogic域中老版本的EJB -->
          <delete file="${weblogic.deploy.dest}/helloEJB.jar"/>
          <!-- 創(chuàng)建META-INF目錄,放置ejb-jar.xml和weblogic-ejb-jar.xml -->
          <mkdir dir="${temp}/ejb_make/META-INF"/>
          <!-- 拷貝ejb-jar.xml和weblogic-ejb-jar.xml 到臨時(shí)目錄-->
          <copy todir="${temp}/ejb_make/META-INF">
          <fileset dir="etc/baseinfo">
          <include name="*.xml"/>
          </fileset>
          </copy>
          <!-- 拷貝所有的helloEJB類到臨時(shí)目錄 -->
          <copy todir="${temp}/ejb_make/">
          <fileset dir="${dest.classes}/"> <!-- dest.classes是輸出的類文件目錄 -->
          <include name="${dest.classes}/helloEJB/**"/>
          </fileset>
          </copy>
          <!-- 將所有這些文件打包成helloEJB.jar -->
          <jar jarfile="${temp}/helloEJB.jar" basedir="${temp}/ejb_make"/>
          <!-- 進(jìn)行weblogic.ejbc編譯 -->
          <java classpath="${wl_cp}" classname="weblogic.ejbc" fork="yes" >
          <classpath>
          <fileset dir="lib">
          <include name="*.jar" />
          </fileset>
          </classpath>
          <arg value="${temp}/helloEJB.jar" />
          <arg value="${temp}/helloEJB_deploy.jar" />
          </java>
          <!-- 拷貝/發(fā)布到WebLogic的{DOMAIN}\applications目錄 -->
          <copy file="${temp}/helloEJB_deploy.jar" todir="${weblogic.deploy.dest}"/>
          </target>


          用Ant配合JUnit實(shí)現(xiàn)單元測(cè)試

          Ant 提供了JUnit任務(wù),可以執(zhí)行單元測(cè)試代碼。如何使用JUnit,以及如何編寫(xiě)測(cè)試用例(TestCase),感興趣的讀者可以參閱JUnit的相關(guān)文檔。在Ant中使用JUnit的方法非常簡(jiǎn)單,首先需要把junit.jar拷入ANT_HOME\lib下,確認(rèn)在這個(gè)目錄下有optional.jar,因?yàn)镴Unit是Ant的擴(kuò)展任務(wù),需要引用這個(gè)擴(kuò)展包。然后就是在Build文件中加入JUnit的任務(wù),代碼如下:

          <target name="run" depends="client">
          <junit printsummary="yes" fork="yes" haltonfailure="yes">
          <classpath>
          <pathelement location="client.jar" />
          </classpath>
          <formatter type="plain" />
          <test name="com.sharetop.antdemo.HelloWorldTest" />
          </junit>
          </target>


          高級(jí)話題

          為Ant開(kāi)發(fā)擴(kuò)展任務(wù)

          為Ant實(shí)現(xiàn)擴(kuò)展任務(wù)其實(shí)是非常容易的,只需按照以下幾個(gè)步驟即可:

          1. 創(chuàng)建一個(gè)Java類繼承org.apache.tools.ant.Task類;

          2. 對(duì)每個(gè)屬性實(shí)現(xiàn)set方法。Ant會(huì)根據(jù)需要自動(dòng)完成類型轉(zhuǎn)換;

          3. 如果擴(kuò)展的任務(wù)需要嵌套其它的Task,那么這個(gè)Java類必需實(shí)現(xiàn)接口org.apache.tools.ant.TaskContainer;

          4. 如果擴(kuò)展的任務(wù)要支持Text,需要增加一個(gè)方法void addText(String);

          5. 對(duì)每個(gè)嵌套的元素,實(shí)現(xiàn)create、add 或 addConfigured 方法;

          6. 實(shí)現(xiàn)public void execute方法;

          7. 在build.xml文件中使用 <taskdef> 來(lái)引用自定義的Task。

          下面以一個(gè)簡(jiǎn)單的例子來(lái)說(shuō)明如何為Ant增加一個(gè)hello任務(wù),它可以連續(xù)打印多條信息,打印的次數(shù)由屬性count指定,而打印的內(nèi)容則由它內(nèi)嵌的一個(gè)helloinfo任務(wù)的message屬性指定,看上去這非常類似JSP中自定義標(biāo)簽的一些概念,實(shí)現(xiàn)代碼如下:

          //HelloInfoTask.java
          package com.sharetop.antdemo;
          import org.apache.tools.ant.*;
          public class HelloInfoTask {
          private String msg;
          public void execute() throws BuildException {
          System.out.println(msg);
          }
          public void setMessage(String msg) {
          this.msg = msg;
          }
          }


          下面是外部Task類的代碼:

          //HelloTask.java
          package com.sharetop.antdemo;
          import org.apache.tools.ant.*;
          public class HelloTask extends Task implements org.apache.tools.ant.TaskContainer
          {
          private Task info;
          private int count;
          public void execute() throws BuildException {
          for(int i=0;i<count;i++)
          info.execute();
          }
          public void setCount(int c){
          this.count=c;
          }
          public void addTask(Task t){
          this.info=t;
          }
          }


          實(shí)現(xiàn)了這兩個(gè)Task,在build.xml文件中定義它的task name,就可以在Target中執(zhí)行它了。如果你不想使用 <taskdef> 標(biāo)簽來(lái)定義Task,也可以通過(guò)修改default.properties文件來(lái)實(shí)現(xiàn)引入新Task,這個(gè)文件位于org.apache.tools.ant.taskdefs 包里。下例是一個(gè)使用 標(biāo)簽來(lái)引入新Task的Build文件部分:

          <target name="hello" depends="client">
          <taskdef name="hello"
          classname="com.sharetop.antdemo.HelloTask" classpath="client.jar"/>
          <taskdef name="helloinfo"
          classname="com.sharetop.antdemo.HelloInfoTask" classpath="client.jar"/>
          <hello count="3" >
          <helloinfo message="hello world" />
          </hello>
          </target>
          posted on 2007-06-19 11:43 麟楓 閱讀(560) 評(píng)論(2)  編輯  收藏 所屬分類: 資料

          Feedback

          # re: Ant教程 2007-07-17 17:47 cylix
          替換文件中的標(biāo)記

          我們將在本節(jié)考察的最后一個(gè)文件系統(tǒng)操作是 replace 任務(wù),它執(zhí)行文件中的查找和替換操作。
          token 屬性指定要查找的字符串,value 屬性指定一個(gè)新的字符串,查找到的標(biāo)記字符串的所有實(shí)
          例都被替換為這個(gè)新的字符串。例如:
          <replace file="input.txt" token="old" value="new"/>
          替換操作將在文件本身之內(nèi)的適當(dāng)位置進(jìn)行。為了提供更詳細(xì)的輸出,可把 summary 屬性設(shè)置為
          true。這將導(dǎo)致該任務(wù)輸出找到和替換的標(biāo)記字符串實(shí)例的數(shù)目。  回復(fù)  更多評(píng)論
            

          # re: Ant教程 2007-07-28 11:40 cylix
          Ant 命令行參考
          從命令行調(diào)用Ant 的語(yǔ)法如下所示:
          ant [option [option...]] [target [target...]]
          option := {-help
          |-projecthelp
          |-version
          |-quiet
          |-verbose
          |-debug
          |-emacs
          |-logfile filename
          |-logger classname
          |-listener classname
          |-buildfile filename
          |-Dproperty=value
          |-find filename}
          語(yǔ)法元素說(shuō)明如下:
          -help
          顯示描述Ant 命令及其選項(xiàng)的幫助信息。
          -projecthelp
          顯示包含在構(gòu)建文件中的、所有用戶編寫(xiě)的幫助文檔。即為各個(gè)<target>
          中description 屬性的文本,以及包含在<description>元素中的任何文
          本。將有description 屬性的目標(biāo)列為主目標(biāo)(“Main target”),沒(méi)有此屬
          性的目標(biāo)則列為子目標(biāo)(“Subtarget”)。
          -version
          要求Ant 顯示其版本信息,然后退出。
          -quiet
          抑制并非由構(gòu)建文件中的echo 任務(wù)所產(chǎn)生的大多數(shù)消息。
          -verbose
          顯示構(gòu)建過(guò)程中每個(gè)操作的詳細(xì)消息。此選項(xiàng)與-debug 選項(xiàng)只能選其一。
          -debug
          顯示Ant 和任務(wù)開(kāi)發(fā)人員已經(jīng)標(biāo)志為調(diào)試消息的消息。此選項(xiàng)與-verbose 只
          能選其一。
          -emacs
          對(duì)日志消息進(jìn)行格式化,使它們能夠很容易地由Emacs 的shell 模式(shellmode)
          所解析;也就是說(shuō),打印任務(wù)事件,但并不縮排,在其之前也沒(méi)有
          [taskname]。
          -logfile filename
          將日志輸出重定向到指定文件。
          -logger classname
          指定一個(gè)類來(lái)處理Ant 的日志記錄。所指定的類必須實(shí)現(xiàn)了org.apache.
          tools.ant.BuildLogger 接口。
          -listener classname
          為Ant 聲明一個(gè)監(jiān)聽(tīng)類,并增加到其監(jiān)聽(tīng)者列表中。在Ant與IDE或其他Java
          程序集成時(shí),此選項(xiàng)非常有用。可以閱讀第六章以了解有關(guān)監(jiān)聽(tīng)者的更多信
          息。必須將所指定的監(jiān)聽(tīng)類編寫(xiě)為可以處理Ant 的構(gòu)建消息接發(fā)。
          -buildfile filename
          指定Ant 需要處理的構(gòu)建文件。默認(rèn)的構(gòu)建文件為build.xml。
          -Dproperty=value
          在命令行上定義一個(gè)特性名-值對(duì)。
          -find filename
          指定Ant 應(yīng)當(dāng)處理的構(gòu)建文件。與-buildfile 選項(xiàng)不同,如果所指定文件在當(dāng)
          前目錄中未找到,-find 就要求Ant 在其父目錄中再進(jìn)行搜索。這種搜索會(huì)繼
          續(xù)在其祖先目錄中進(jìn)行,直至達(dá)到文件系統(tǒng)的根為止,在此如果文件還未找
          到,則構(gòu)建失敗。
          構(gòu)建文件輪廓
          下面是一個(gè)通用的構(gòu)建文件,它很適合作為一個(gè)模板。構(gòu)建文件包括<project>
          元素,以及其中嵌套的<target>、<property> 和<path> 元素。
          <project default="all">
          <property name="a.property" value="a value"/>
          <property name="b.property" value="b value"/>
          <path id="a.path">
          <pathelement location="${java.home}/jre/lib/rt.jar"/>
          </path>
          <target name="all">
          <javac srcdir=".">
          <classpath refid="a.path"/>
          </javac>
          </target>
          </project>

          關(guān)于構(gòu)建文件有幾點(diǎn)需要記住:
          ● 所有構(gòu)建文件都要有<project>元素,而且至少有一個(gè)<target> 元素。
          ● 對(duì)于<project> 元素的default 屬性并沒(méi)有默認(rèn)值。
          ● 構(gòu)建文件并不一定要被命名為build.xml。不過(guò)build.xml 是Ant 要搜索的默
          認(rèn)文件名。
          ● 每個(gè)構(gòu)建文件只能有一個(gè)<project> 元素。  回復(fù)  更多評(píng)論
            


          只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 平陆县| 四会市| 邳州市| 扎赉特旗| 垦利县| 龙南县| 台中市| 蒲江县| 嘉祥县| 吉林省| 景洪市| 游戏| 靖江市| 平湖市| 夏邑县| 海南省| 维西| 印江| 和林格尔县| 喀什市| 老河口市| 武胜县| 岳池县| 米易县| 罗定市| 禄劝| 凤庆县| 措勤县| 柘荣县| 临漳县| 芦山县| 乌兰察布市| 馆陶县| 和田县| 兴山县| 大荔县| 靖安县| 文昌市| 日土县| 黑龙江省| 楚雄市|