jk自慰喷水,国产又粗又黄又猛又爽高潮视频,国内伦理一级伦理麻豆,亚洲成人91

哈爾濱今日靈圖

[其他技能培訓(xùn)]
獵學(xué)網(wǎng)訂閱號(hào)
獵學(xué)網(wǎng)官方企業(yè)微信
位置: 獵學(xué)網(wǎng) > 學(xué)校機(jī)構(gòu) > 哈爾濱今日靈圖 > 學(xué)習(xí)資訊> 手把手教你用 Java 實(shí)現(xiàn) AOP

手把手教你用 Java 實(shí)現(xiàn) AOP

118 2017-04-19

介紹

眾所周知,AOP(面向切面編程)是Spring框架的特色功能之一。通過設(shè)置橫切關(guān)注點(diǎn)(crosscuttingconcerns),AOP提供了極高的擴(kuò)展性。那AOP在Spring中是怎樣運(yùn)作的呢?當(dāng)你只能使用corejava,卻需要AOP技術(shù)時(shí),這個(gè)問題的解答變得極為關(guān)鍵。不僅如此,在高級技術(shù)崗位的面試中,此類問題也常作為考題出現(xiàn)。這不,我的朋友最近參加了一個(gè)面試,就被問到了這樣一個(gè)棘手的問題——如何在不使用Spring及相關(guān)庫,只用coreJava的條件下實(shí)現(xiàn)AOP。因此,我將在本文中提供一份大綱,幫助大家了解如何只用coreJava實(shí)現(xiàn)一個(gè)AOP(當(dāng)然啦,這種AOP在功能上有一定的局限性)。注意,本文不是一篇有關(guān)SpringAOP與JavaAOP的對比研究,而是有關(guān)在coreJava中借助固有的設(shè)計(jì)模式實(shí)現(xiàn)AOP的教程。

想必讀者已經(jīng)知道AOP是什么,也知道在Spring框架中如何使用它,因此本文只著眼于如何在不用Spring的前提下實(shí)現(xiàn)AOP。首先,我們得知道,Spring是借助了JDKproxy和CGlib兩種技術(shù)實(shí)現(xiàn)AOP的。JDKdynamicproxy提供了一種靈活的方式來hook一個(gè)方法并執(zhí)行指定的操作,但執(zhí)行操作時(shí)得有一個(gè)限制條件:必須先提供一個(gè)相關(guān)的接口以及該接口的實(shí)現(xiàn)類。實(shí)踐出真知,讓我們透過一個(gè)案例來理解這句吧!現(xiàn)在有一個(gè)計(jì)算器程序,用于完成一些數(shù)學(xué)運(yùn)算。讓我們來考慮下除法功能,此時(shí)的問題是:如果coreframework已經(jīng)具備了一份實(shí)現(xiàn)除法的代碼,我們能否在代碼執(zhí)行時(shí)劫持(highjack)它并執(zhí)行額外的校驗(yàn)?zāi)??答案是肯定的,我將用下面提供的代碼片段來證明這點(diǎn)。首先來看基礎(chǔ)接口的代碼:

publicinterfaceCalculator{

publicintcalculate(inta,intb);

}

該接口實(shí)現(xiàn)類的代碼如下:

publicclassCalculatorImplimplementsCalculator{

@Override

publicintcalculate(inta,intb){

returna/b;

}

}

假設(shè)我們既不能修該上面的代碼,也不能對核心庫進(jìn)行任何改動(dòng),怎樣才能完美地實(shí)現(xiàn)校驗(yàn)功能呢?不如試下JDKdynamicproxy的功能吧。

publicclassSomeHandlerimplementsInvocationHandler{

//Codeomittedforsimplicity…..

@Override

publicObjectinvoke(Objectproxy,Methodmethod,Object[]params)throwsThrowable{

//Yourplexbusinessvalidationandlogic

Objectresult=method.invoke(targetObject,params);

returnresult;

}

}

讓我們通過測試類來看看由JDKdynamicproxy實(shí)現(xiàn)的校驗(yàn)功能的效果如何。

publicstaticvoidmain(String[]args){

CalculatorImplcalcImpl=newCalculatorImpl();

Calculatorproxied=(Calculator)ProxyFactory.getProxy(Calculator.class,calcImpl,

newSomeHandler(calcImpl));

intresult=proxied.calculate(20,10);

System.out.println("FInalResult:::"+result);

}

從結(jié)果可以看出,簡單地實(shí)現(xiàn)功能強(qiáng)大的InvocationHandler接口,我們便能得到一個(gè)hookingimplementation。按照J(rèn)DK文檔的描述,InvocationHandler接口是借助一個(gè)代理實(shí)例(proxyinstance)來處理一個(gè)方法調(diào)用的。

現(xiàn)在我們已經(jīng)知道,InvocationHandler的invoke()方法能夠幫助我們解決問題。那么再來解決一個(gè)新問題——怎樣才能在方法執(zhí)行的前后執(zhí)行操作呢?說的更具體一些,我們能通過添加多個(gè)aop(before、after、around)來hook一個(gè)方法嗎(譯注:原文為addmultipleaops,但我認(rèn)為Handler是充當(dāng)Aspect的角色)?答案同樣是肯定的。按照以下的步驟建立一個(gè)精簡的代碼模板便能滿足這樣的需求:

1.

創(chuàng)建一個(gè)抽象類,用于將aop應(yīng)用于目標(biāo)對象上。

2.

創(chuàng)建名為BeforeHandler和AfterHandler的兩個(gè)aop。前者在方法執(zhí)行之前工作,而后者則在方法執(zhí)行結(jié)束后工作。

3.

創(chuàng)建一個(gè)代理類,使所有的aophandler和目標(biāo)對象只需作為參數(shù)傳入,就能創(chuàng)建一個(gè)hook。

4.

加入你自己的業(yè)務(wù)邏輯或者橫切關(guān)注點(diǎn)。

5.

最后,通過傳入相關(guān)的參數(shù)創(chuàng)建代理對象(proxyobject)。

技術(shù)實(shí)現(xiàn)概要

(譯注:此處是核心代碼片段,如果想運(yùn)行該實(shí)例,需進(jìn)入下方提供的鏈接下載完整代碼)

創(chuàng)建一個(gè)handler的抽象類:

publicabstractclassAbstractHandlerimplementsInvocationHandler{

privateObjecttargetObject;

publicvoidsetTargetObject(ObjecttargetObject){

this.targetObject=targetObject;

}

}

創(chuàng)建名為BeforeHandler和AfterHandler的兩個(gè)易擴(kuò)展的handler抽象類:

publicabstractclassBeforeHandlerextendsAbstractHandler{

publicabstractvoidhandleBefore(Objectproxy,Methodmethod,Object[]args);

publicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{

handleBefore(proxy,method,args);

returnmethod.invoke(getTargetObject(),args);

}

}

publicabstractclassAfterHandlerextendsAbstractHandler{

publicabstractvoidhandleAfter(Objectproxy,Methodmethod,Object[]args);

publicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{

Objectresult=method.invoke(getTargetObject(),args);

handleAfter(proxy,method,args);

returnresult;

}

}

創(chuàng)建Proxy的工廠類:

publicclassProxyFactory{

publicstaticObjectgetProxy(ObjecttargetObject,

Listhandlers){

//Codetogettheproxy

returnproxyObject;

}else{

returntargetObject;

}

}

}

以下為測試代碼:

CalculatorImplcalcImpl=newCalculatorImpl();

BeforeHandlerbefore=newBeforeHandlerImpl();

AfterHandlerafter=newAfterHandlerImpl();

Listhandlers=newArrayList();

handlers.add(before);

handlers.add(after);

Calculatorproxy=(Calculator)ProxyFactory.getProxy(calcImpl,

handlers);

intresult=proxy.calculate(20,10);

配置

以上的代碼片段簡明扼要地解釋了AOP在結(jié)構(gòu)上的實(shí)現(xiàn)(structuralimplementation)。當(dāng)然,如果能通過實(shí)際的測試將其運(yùn)用到現(xiàn)實(shí)中去,那就再好不過了。讀者可在下面的鏈接中獲取完整的工程文件,并在Java編輯器中配置它們,最后通過其中的測試類來檢驗(yàn)效果。

總結(jié)

希望這篇簡短的有關(guān)AOP文章能夠幫助到大家。需說明的是,本文只實(shí)現(xiàn)了before和after兩種aop,而另外兩種,即“Around”和“Throw”,則希望讀者自行完成。

請聯(lián)系網(wǎng)站客服,了解詳細(xì)的優(yōu)惠課程信息~

優(yōu)質(zhì)、權(quán)威、便捷、省心

溫馨提示: 專業(yè)老師1對1為您解答    馬上填寫,¥1000 元豪禮免費(fèi)領(lǐng)!

掃一掃
獲取更多福利

×
獵學(xué)網(wǎng)