主頁(yè) > 知識(shí)庫(kù) > Java中基于Aspectwerkz的AOP

Java中基于Aspectwerkz的AOP

熱門(mén)標(biāo)簽:騰訊地圖標(biāo)注商戶(hù)改名注冊(cè)入駐 電話機(jī)器人的特色和創(chuàng)新 商丘百應(yīng)電話機(jī)器人有沒(méi)有效果 開(kāi)封便宜外呼系統(tǒng)報(bào)價(jià) 淮南騰訊地圖標(biāo)注 漯河辦理400電話 怎樣把地圖標(biāo)注出來(lái) 黃石智能營(yíng)銷(xiāo)電銷(xiāo)機(jī)器人效果 地圖標(biāo)注人員兼職

    一、AOP編程概覽

  面向?qū)ο缶幊碳夹g(shù)進(jìn)入軟件開(kāi)發(fā)的主流對(duì)軟件的開(kāi)發(fā)方式產(chǎn)生了極大的影響,開(kāi)發(fā)者可以用一組實(shí)體以及這些實(shí)體之間的關(guān)系將系統(tǒng)形象地表示出來(lái),這使得他們能夠設(shè)計(jì)出規(guī)模更大、更復(fù)雜的系統(tǒng),開(kāi)發(fā)周期也比以前更短。OO開(kāi)發(fā)的唯一問(wèn)題是,它本質(zhì)上是靜態(tài)的,需求的細(xì)微變化就可能對(duì)開(kāi)發(fā)進(jìn)度造成重大影響。

  Aspect-Oriented Programming(AOP)是對(duì)OO技術(shù)的補(bǔ)充和完善,它允許開(kāi)發(fā)者動(dòng)態(tài)地修改靜態(tài)的OO模型,構(gòu)造出一個(gè)能夠不斷增長(zhǎng)以滿足新增需求的系統(tǒng),就象現(xiàn)實(shí)世界中的對(duì)象會(huì)在其生命周期中不斷改變自身,應(yīng)用程序也可以在發(fā)展中擁有新的功能。

  例如,許多人想必有過(guò)在開(kāi)發(fā)簡(jiǎn)單的Web應(yīng)用時(shí)將Servlet作為入口點(diǎn)的經(jīng)驗(yàn),即用Servlet接收HTML表單的輸入,經(jīng)過(guò)處理后返回給用戶(hù)。開(kāi)始時(shí)的Servlet可能是非常簡(jiǎn)單的,只有剛好滿足用戶(hù)需求的最少量的代碼。然而,隨著“第二需求”的實(shí)現(xiàn),例如實(shí)現(xiàn)異常處理、安全、日志等功能,代碼的體積就會(huì)增加到原來(lái)的三、四倍——之所以稱(chēng)之為“第二需求”,是因?yàn)镾ervlet的基本功能是接受和處理用戶(hù)的請(qǐng)求,對(duì)于這個(gè)目標(biāo)來(lái)說(shuō),日志、安全之類(lèi)的機(jī)制并不是必不可少的。

  AOP允許動(dòng)態(tài)地改變OO的靜態(tài)模型,不必修改原來(lái)的靜態(tài)模型也可以加入滿足第二需求所需的代碼(實(shí)際上,甚至連原來(lái)的源代碼也不需要)。更令人稱(chēng)奇的是,后來(lái)加入的代碼往往可以集中在一個(gè)地方,而不必象單純使用OO時(shí)那樣將后來(lái)加入的代碼分散到整個(gè)模型。

  二、基本術(shù)語(yǔ)

  在介紹AOP開(kāi)發(fā)實(shí)例之前,我們先來(lái)了解幾個(gè)標(biāo)準(zhǔn)的AOP術(shù)語(yǔ),以便更好地掌握相關(guān)的概念。

   Cross-cutting concern

  在OO模型中,雖然大部份的類(lèi)只有單一的、特定的功能,但它們通常會(huì)與其他類(lèi)有著共同的第二需求。例如,當(dāng)線程進(jìn)入或離開(kāi)某個(gè)方法時(shí),我們可能既要在數(shù)據(jù)訪問(wèn)層的類(lèi)中記錄日志,又要在UI層的類(lèi)中記錄日志。雖然每個(gè)類(lèi)的基本功能極然不同,但用來(lái)滿足第二需求的代碼卻基本相同。

   Advice

  它是指想要應(yīng)用到現(xiàn)有模型的附加代碼。在本例中,它是指線程進(jìn)入或退出某個(gè)方法時(shí)要運(yùn)行的日志代碼。

   Point-cut

  這個(gè)術(shù)語(yǔ)是指應(yīng)用程序中的一個(gè)執(zhí)行點(diǎn),在這個(gè)執(zhí)行點(diǎn)上需要采用前面的cross-cutting concern。在本例中,當(dāng)線程進(jìn)入一個(gè)方法時(shí)出現(xiàn)一個(gè)Point-cut,當(dāng)線程離開(kāi)方法時(shí)又出現(xiàn)另一個(gè)Point-cut。

   Aspect

  Point-cut和advice結(jié)合在一起就叫做aspect。在下面的例子中,我們通過(guò)定義一個(gè)point-cut并給予適當(dāng)?shù)腶dvice加入了一個(gè)日志(logging)aspect。

  AOP還有其它許多特性和術(shù)語(yǔ),例如引入(Introduction),即把接口/方法/域引入到現(xiàn)有的類(lèi)——它極大地拓寬了開(kāi)發(fā)者的想象力。不過(guò)本文只介紹一些最基本的持性,熟悉這里介紹的概念后,你再深入一步研究AOP的其它特性,看看如何在自己的開(kāi)發(fā)環(huán)境中使用它們。

  三、現(xiàn)有的框架

  目前最成熟、功能最豐富的AOP框架當(dāng)數(shù)AspectJ,AspectJ已成為大多數(shù)其它框架跟從的標(biāo)準(zhǔn)。但是,AspectJ也走出了非同尋常的一步,它的實(shí)現(xiàn)為Java語(yǔ)言增添了新的關(guān)鍵詞。雖然新的語(yǔ)法并不難學(xué),但卻意味著我們必須換一個(gè)編譯器,還要重新配制編輯器,只有這樣才能適應(yīng)新的語(yǔ)法。在規(guī)模較大的開(kāi)發(fā)組中,這些要求可能難以辦到,因?yàn)檎麄€(gè)開(kāi)發(fā)小組都會(huì)受到影響。由于語(yǔ)言本身的變化,開(kāi)發(fā)小組把AOP技術(shù)引入到現(xiàn)有項(xiàng)目的學(xué)習(xí)周期隨之延長(zhǎng)。

  現(xiàn)在我們需要的是這樣一個(gè)框架,它可以方便地引入,且不會(huì)對(duì)原來(lái)的開(kāi)發(fā)和構(gòu)造過(guò)程產(chǎn)生任何影響。滿足這些要求的框架不止一個(gè),例如JBoss AOP、Nanning、Aspectwerkz(AW)。本文選用的是Aspectwerkz,因?yàn)樗赡苁亲钊菀讓W(xué)習(xí)的框架,也是最容易集成到現(xiàn)有項(xiàng)目的框架。

  Aspectwerkz由Jonas Boner和Alexandre Vasseur創(chuàng)建,它是目前最快速、功能最豐富的框架之一。雖然它還缺乏AspectJ的某些功能,但己足以滿足大多數(shù)開(kāi)發(fā)者在許多情形下的需要。

  Aspectwerkz最令人感興趣的特性之一是它能夠以?xún)煞N不同的模式運(yùn)行:聯(lián)機(jī)模式和脫機(jī)模式。在聯(lián)機(jī)模式下,AW直接干預(yù)屬于JVM的底層類(lèi)裝入機(jī)制,截取所有的類(lèi)裝入請(qǐng)求,對(duì)字節(jié)碼實(shí)施即時(shí)轉(zhuǎn)換。AW提供了干預(yù)類(lèi)裝入過(guò)程的許多選項(xiàng),另外還有一個(gè)替代bin/java命令的封裝腳本,這個(gè)腳本能夠根據(jù)Java版本和JVM能力自動(dòng)生成一組可運(yùn)行的配制。對(duì)于開(kāi)發(fā)者,聯(lián)機(jī)模式有許多優(yōu)點(diǎn),它能插入到任何類(lèi)裝入器并在類(lèi)裝入期間生成新的類(lèi)。也就是說(shuō),我們不必手工修改應(yīng)用程序的類(lèi),只要按通常的方式部署即可。不過(guò),聯(lián)機(jī)模式要求對(duì)應(yīng)用服務(wù)器進(jìn)行額外的配制,有時(shí)這一要求可能很難滿足。

  在脫機(jī)模式下,生成類(lèi)需要二個(gè)步驟。第一步是用標(biāo)準(zhǔn)的編譯器編譯,第二步是重點(diǎn)——以脫機(jī)模式運(yùn)行AWcompiler編譯器,讓它處理新生成的類(lèi)。編譯器將修改這些類(lèi)的字節(jié)碼,根據(jù)一個(gè)XML文件的定義,在適當(dāng)?shù)膒oint-cut插入advice。脫機(jī)模式的優(yōu)點(diǎn)是AWcompiler生成的類(lèi)能夠在任何JVM 1.3以上的虛擬機(jī)運(yùn)行,本文下面要用的就是這種模式,因?yàn)樗恍枰獙?duì)Tomcat作任何修改,只要對(duì)構(gòu)造過(guò)程稍作修改就可以照搬到大多數(shù)現(xiàn)有的項(xiàng)目。

  四、安裝

  本文將以一個(gè)簡(jiǎn)單的Web應(yīng)用程序?yàn)槔?,它用Ant編譯,部署在Tomcat 4+ Servlet容器上。下面我們假定讀者己準(zhǔn)備好上述環(huán)境,包括JVM 1.3+,同時(shí)Tomcat被設(shè)置成從webapps文件夾自動(dòng)部署應(yīng)用,自動(dòng)將WAR擴(kuò)展到目錄(這是Tomcat默認(rèn)的操作方式,因此只要你尚未修改Tomcat的運(yùn)行方式,下面的范例可直接運(yùn)行)。我們將把Tomcat的安裝位置稱(chēng)為%TOMCAT_HOME%。

 ?、?從http://apectwerkz.codehaus.org/下載Aspectwerkz,解開(kāi)壓縮到適當(dāng)?shù)奈恢?。我們將把這個(gè)位置稱(chēng)為%ASPECTWERKZ_HOME%。

 ?、?設(shè)置%ASPECTWERKZ_HOME%環(huán)境變量。

  ⑶ 將Aspectwerkz加入到PATH環(huán)境變量,即設(shè)置set PATH=%PATH%;%ASPECTWERKZ_HOME%inaspectwerkz

  ⑷ 下載本文的示范程序,將它放入%TOMCAT_HOME%webapps文件夾。

 ?、?將Aspectwerkz的運(yùn)行時(shí)類(lèi)加入到Tomcat的classpath。你可以將它的JAR文件放入示例應(yīng)用的WEB-INFlib文件夾,或放入%TOMCAT_HOME%commonlib。

    五、編譯示例應(yīng)用

  如果你想深入研究一下本文的示例應(yīng)用,可以解開(kāi)WAR文件提取它的內(nèi)容。你會(huì)發(fā)現(xiàn)根目錄下有一個(gè)aspectwerkz.xml文件,構(gòu)造應(yīng)用時(shí)它會(huì)被復(fù)制到WEB-INF/classes目錄。Servlet和advice的源文件在WEB-INF/src目錄下,另外還有一個(gè)構(gòu)建這些類(lèi)的ANT腳本。

  在運(yùn)行這個(gè)示例程序之前,你還要對(duì)它進(jìn)行后期編譯。下面是具體的操作步驟:

 ?、?在命令行窗口中,轉(zhuǎn)到解開(kāi)WAR文件的目錄。

 ?、?輸入下面的命令調(diào)用AW編譯器:aspectwerkz -offline aspectwerkz.xml WEB-INF/classes -cp %TOMCAT_HOME%commonlibservlet.jar。如后期編譯順利通過(guò),應(yīng)看到下面的輸出:

  ( 1 s )

  SUCCESS: WEB-INFclasses

  在構(gòu)建文件中有一個(gè)名稱(chēng)為war的ANT任務(wù),你可以用它重新創(chuàng)建WAR文件。

  六、運(yùn)行示例應(yīng)用

  首先啟動(dòng)(或重新啟動(dòng))Tomcat,然后在瀏覽器中打開(kāi)http://localhost:8080/demo/。

  頁(yè)面打開(kāi)后,可以看到一個(gè)帶二個(gè)輸入框的HTML表單,一個(gè)輸入名字,一個(gè)輸入郵件地址。輸入一些數(shù)據(jù),然后點(diǎn)擊按鈕提交表單,出現(xiàn)一個(gè)頁(yè)面顯示出聯(lián)系人信息和一個(gè)指向聯(lián)系人清單的鏈接。

  七、代碼分析

  JSP頁(yè)面就不分析了,現(xiàn)在我們對(duì)它不感興趣。我們來(lái)看看AOPServlet的代碼。

  package example;

import java.io.*;
import javax.servlet.*;
import javax.servlet.http.*;
public class AOPServlet extends HttpServlet {
 public void doGet(HttpServletRequest request, HttpServletResponse response)
 throws ServletException, IOException {
  Person person = new Person();
  if (request.getParameter("name") != null) {
   person.setName(
   request.getParameter("name"));
  }
  if (request.getParameter("email") != null) {
   person.setEmail(
   request.getParameter("email"));
  }
  request.setAttribute("person", person);
  RequestDispatcher rd =request.getRequestDispatcher("/view.jsp");
  rd.forward(request, response);
 }
}

  在這個(gè)例子中,Servlet的代碼己盡量精簡(jiǎn),只包含一些必不可少的代碼,如創(chuàng)建了一個(gè)綁定請(qǐng)求參數(shù)的對(duì)象等,但沒(méi)有持久化操作,不需要額外的imports,它只實(shí)現(xiàn)了作為Servlet必須實(shí)現(xiàn)的最基本的操作。

  然而,根據(jù)說(shuō)明文檔的要求,這個(gè)應(yīng)用程序必須將所有Person類(lèi)型的對(duì)象特久化,所以要為這個(gè)應(yīng)用程序加入一個(gè)aspect。為創(chuàng)建這個(gè)aspect,我們首先要?jiǎng)?chuàng)建一個(gè)aspectwerkz.xml文件并將該文件放入classpath指定的目錄。本文示例提供了一個(gè)簡(jiǎn)單的例子,你可以用編輯器打開(kāi)查看。

  aspectwerkz.xml的第一部份定義了可用的advice,我們可以根據(jù)需要加入任意數(shù)量的advice:

 ?。糰dvice-def name="persist" class="example.PersistenceAdvice" deployment-model="perJVM"/>

  在這個(gè)片段中,我們定義了一個(gè)名稱(chēng)為persist的advice,它的類(lèi)型是example.PersistenceAdvice。最后一個(gè)屬性定義了該advice的排它性,在這里它的值是perJVM,表示在每一個(gè)JVM中只創(chuàng)建該advice的一個(gè)實(shí)例(有關(guān)部署模式的更多說(shuō)明,請(qǐng)參見(jiàn)Aspectwerkz的文檔。

  第二部份開(kāi)始定義aspect,這里就是我們將advice映射到point-cut創(chuàng)建aspect的地方。

  <aspect name="servlet">
 ?。紁ointcut-def name="all" type="method"
  pattern="* example.*Servlet.doGet(..)"/>
 ?。糱ind-advice pointcut="all">
  <advice-ref name="persist"/>
 ?。?bind-advice>
 ?。?aspect>

  下面我們一行一行地分析這段代碼:

 ?、?我們創(chuàng)建了一個(gè)叫做servlet的aspect。如有必要,我們可以創(chuàng)建任意數(shù)量的aspect。

  ⑵ 在第二行,我們創(chuàng)建了一個(gè)叫做all的point-cut,它只適用于方法(type="method")。

 ?、?第三行我們用一個(gè)正則表達(dá)式規(guī)定了把a(bǔ)dvice應(yīng)用到哪里。在這個(gè)例子中,我們指出應(yīng)用advice的條件是:不管返回值的類(lèi)型是什么(第一個(gè)“*”),名稱(chēng)以servlet結(jié)尾(*servlet)且包含一個(gè)帶任意參數(shù)的doGet方法(doGet(..))的example包里面的類(lèi)。

 ?、?在第四行,我們告訴Aspectwerkz編譯器要把后面的advice應(yīng)用到所有的point-cut。

 ?、?在這里我們聲明要使用的advice是persist。

  現(xiàn)在我們知道了如何映射point-cut與advice創(chuàng)建出aspect,下面來(lái)看看一個(gè)提供advice的類(lèi)的實(shí)例。在映射文件中,我們注冊(cè)了一個(gè)example.PersistenceAdvice類(lèi)型的advice,下面是該類(lèi)型的源代碼:

  package example;

import javax.servlet.http.*;
import org.codehaus.aspectwerkz.advice.*;
import org.codehaus.aspectwerkz.joinpoint.*;

public class PersistenceAdvice extends AroundAdvice {
 public PersistenceAdvice() {
  super();
 }
 public Object execute(final JoinPoint joinPoint)
 throws Throwable {
  MethodJoinPoint jp =(MethodJoinPoint) joinPoint;
  final Object result = joinPoint.proceed();
  Object[] parameters = jp.getParameters();
  if (parameters[0] instanceof HttpServletRequest) {
   HttpServletRequest request =(HttpServletRequest) parameters[0];
   if (request.getAttribute("person") != null) {
    Person contact =(Person) request.getAttribute("person");
    ContactManager persistent = new ContactManager();
    String fileName =(request.getRealPath("/")+"contacts.txt");
    persistent.save(contact, fileName);
   }
  }
  return result;
 }
}


  execute()方法的第一行很容易理解,就是盡量把它定型成最具體的類(lèi)型,第二行或許是最重要的:因?yàn)槲覀兿胍\(yùn)行該方法并檢查結(jié)果,所以必須調(diào)用proceed()。在下一部份,我們捕獲HttpServletRequest,提取由Servlet放入的對(duì)象(記住,此時(shí)doGet()方法己運(yùn)行結(jié)束)。

  最后,我們創(chuàng)建一個(gè)名稱(chēng)為ContactManager的類(lèi),它的功能是把Person的數(shù)據(jù)保存到一個(gè)文本文件。實(shí)際上,要把數(shù)據(jù)保存到XML文件、數(shù)據(jù)庫(kù)或其它持久化存儲(chǔ)機(jī)制也很方便。

  這里需要掌握的一點(diǎn)是,在設(shè)計(jì)應(yīng)用或建立原型的階段,Servlet并不知道未來(lái)會(huì)發(fā)生什么變化,第二階段的功能可以隨時(shí)加入,正因?yàn)槿绱?,所以我們說(shuō)應(yīng)用程序能夠在發(fā)展過(guò)程中學(xué)習(xí)新的能力,以后要添加新的功能非常方便。

  【結(jié)束語(yǔ)】 我們?cè)谇懊娴睦又性囼?yàn)了一個(gè)簡(jiǎn)單的應(yīng)用,將它部署到Tomcat,并用瀏覽器運(yùn)行和測(cè)試它的功能。雖然這個(gè)應(yīng)用本身并無(wú)任何實(shí)際用途,但它示范和證實(shí)了一些非常有用的概念。想象一下,你將可以快速地建立原型,完成后再引入安全、日志、持久化、緩沖之類(lèi)的Cross-cutting concern。不管原始應(yīng)用的規(guī)模有多大,你將能夠在十分鐘之內(nèi)輕松地為整個(gè)應(yīng)用加入日志功能!

  希望你能夠超越本文的簡(jiǎn)單例子,去看看如何在自己的項(xiàng)目中采用AOP技術(shù)。熟悉AOP的概念當(dāng)然需要一定的時(shí)間,但肯定會(huì)得到回報(bào),對(duì)于一個(gè)中等規(guī)模的項(xiàng)目,它會(huì)讓你省下數(shù)星期時(shí)間,或者少寫(xiě)數(shù)千行重復(fù)的代碼。

 

您可能感興趣的文章:
  • 詳解使用Java原生代理實(shí)現(xiàn)AOP實(shí)例
  • Java實(shí)現(xiàn)AOP面向切面編程的實(shí)例教程
  • 實(shí)例講解Java的Spring框架中的AOP實(shí)現(xiàn)
  • Java的Spring框架中AOP項(xiàng)目的一般配置和部署教程
  • 舉例講解Java的Spring框架中AOP程序設(shè)計(jì)方式的使用
  • Java的Spring框架下的AOP編程模式示例
  • java使用動(dòng)態(tài)代理來(lái)實(shí)現(xiàn)AOP(日志記錄)的實(shí)例代碼
  • Java動(dòng)態(tài)代理實(shí)現(xiàn)AOP
  • 體驗(yàn)Java 1.5中面向(AOP)編程
  • Java AOP知識(shí)詳細(xì)介紹

標(biāo)簽:亳州 拉薩 岳陽(yáng) 大興安嶺 馬鞍山 鄭州 紅河 武威

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Java中基于Aspectwerkz的AOP》,本文關(guān)鍵詞  Java,中,基于,Aspectwerkz,的,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Java中基于Aspectwerkz的AOP》相關(guān)的同類(lèi)信息!
  • 本頁(yè)收集關(guān)于Java中基于Aspectwerkz的AOP的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章