本節(jié)要點(diǎn):
企業(yè)建站必須是能夠以充分展現(xiàn)企業(yè)形象為主要目的,是企業(yè)文化與產(chǎn)品對(duì)外擴(kuò)展宣傳的重要窗口,一個(gè)合格的網(wǎng)站不僅僅能為公司帶來(lái)巨大的互聯(lián)網(wǎng)上的收集和信息發(fā)布平臺(tái),創(chuàng)新互聯(lián)面向各種領(lǐng)域:酒店設(shè)計(jì)等成都網(wǎng)站設(shè)計(jì)、網(wǎng)絡(luò)營(yíng)銷推廣解決方案、網(wǎng)站設(shè)計(jì)等建站排名服務(wù)。
Java靜態(tài)代理
Jdk動(dòng)態(tài)代理
1 面向?qū)ο笤O(shè)計(jì)思想遇到的問題
在傳統(tǒng)OOP編程里以對(duì)象為核心,并通過對(duì)象之間的協(xié)作來(lái)形成一個(gè)完整的軟件功能,由于對(duì)象可以繼承,因此我們可以把具有相同功能或相同特征的屬性抽象到一個(gè)層次分明的類結(jié)構(gòu)體系中。隨著軟件規(guī)范的不斷擴(kuò)大,專業(yè)化分工越來(lái)越系列,以及OOP應(yīng)用實(shí)踐的不斷增多,隨之也暴露了一些OOP無(wú)法很好解決的問題。
現(xiàn)在假設(shè)系統(tǒng)中有三段完全相似的代碼,這些代碼通常會(huì)采用“復(fù)制”、“粘貼”方式來(lái)完成,通過這種方式開發(fā)出來(lái)的軟件如圖所示:
可能讀者已經(jīng)發(fā)現(xiàn)了這種做法的不足之處,如果有一天,藍(lán)色背景的代碼需要修改,那是不是要同時(shí)修改三個(gè)地方?如果不僅僅是這三個(gè)地方包含這段代碼,而是100個(gè),甚至是1000個(gè)地方,那會(huì)是什么后果?
記錄日志在代碼中無(wú)處不在---先來(lái)看一個(gè)例子:
為了跟蹤應(yīng)用程序的運(yùn)行過程,很多方法都需要記錄日志信息。我們一般這樣寫:
//log4j的使用見文章“l(fā)og4j介紹” import org.apache.log4j.Logger; public class Person { private Logger logger = Logger.getLogger(Person.class); public void sleep(){ logger.info(“開始執(zhí)行時(shí)間:“ + new Date()); System.out.println("睡覺中"); logger.info(“執(zhí)行結(jié)束時(shí)間:” + new Date()); } public void eating(){ logger.info("開始執(zhí)行時(shí)間:“ + new Date()"); System.out.println("正在吃飯中"); logger.info("“執(zhí)行結(jié)束時(shí)間:” + new Date()"); } }
提問:弊端在哪里?
l混淆了業(yè)務(wù)方法本身的職責(zé)
l維護(hù)工作量巨大
2解決方案1
靜態(tài)代理:
1、需要知道核心類(被代理類)是哪一個(gè)類,并且有什么方法?!?br /> 2、非核心的代碼需要重復(fù)寫多次,顯得代碼的結(jié)構(gòu)臃腫,形成代碼冗余。
3、非核心類(代理類)需要實(shí)現(xiàn)核心類(被代理類)實(shí)現(xiàn)的接口,也就是他們需要實(shí)現(xiàn)共同的接口,但是以核心類實(shí)現(xiàn)的接口(被代理類)為準(zhǔn)。
l目地是將業(yè)務(wù)代碼與日志代碼完全分離,實(shí)現(xiàn)松散耦合.
l代理對(duì)象與被代理對(duì)象必須實(shí)現(xiàn)同一接口,在代理對(duì)象中實(shí)現(xiàn)與日志記錄的相關(guān)服務(wù),并在需要的時(shí)候呼叫被代理對(duì)象,而被代理對(duì)象只保留業(yè)務(wù)代碼.
靜態(tài)代理的實(shí)現(xiàn)
1)定義接口:
public interface IPerson { public abstract void sleep(); public abstract void eating(); }
2) 被代理類
public class Person implements IPerson { public void sleep(){ System.out.println("睡覺中"); } public void eating(){ System.out.println("正在吃飯中"); } }
3) 代理類
import org.apache.log4j.Logger; public class PersonProxy implements IPerson { private IPerson person; private Logger logger = Logger.getLogger(PersonProxy.class); public PersonProxy(IPerson person) { this.person = person; } public void eating() { logger.info("開始執(zhí)行時(shí)間:“ + new Date()"); person.eating(); logger.info("“執(zhí)行結(jié)束時(shí)間:” + new Date()"); } public void sleep() { logger.info("開始執(zhí)行時(shí)間:“ + new Date()"); person.sleep(); logger.info("“執(zhí)行結(jié)束時(shí)間:” + new Date()"); } }
4) 測(cè)試類
package com.aptech.aop2; public class PersonTest { public static void main(String[] args) { IPerson proxy = new PersonProxy(new Person()); proxy.eating(); proxy.sleep(); } }
靜態(tài)代理的弊端:
一個(gè)代理接口只能服務(wù)于一種類型的對(duì)象.對(duì)于稍大點(diǎn)的項(xiàng)目根本無(wú)法勝任.
3 解決方案2-動(dòng)態(tài)代理
InvocationHandler:每一個(gè)動(dòng)態(tài)代理類都必須實(shí)現(xiàn)InvocationHandler這個(gè)接口,并且每個(gè)代理類的實(shí)例都關(guān)聯(lián)到了一個(gè)handler,當(dāng)我們通過代理對(duì)象調(diào)用一個(gè)方法的時(shí)候,這個(gè)方法的調(diào)用就會(huì)被轉(zhuǎn)發(fā)為由InvocationHandler這個(gè)接口的invoke方法來(lái)進(jìn)行調(diào)用。
在JDK1.3之后加入了可協(xié)助開發(fā)的動(dòng)態(tài)代理功能.不必為特定對(duì)象與方法編寫特定的代理對(duì)象,使用動(dòng)態(tài)代理,可以使得一個(gè)處理者(Handler)服務(wù)于各個(gè)對(duì)象.
一個(gè)處理者的類設(shè)計(jì)必須實(shí)現(xiàn)java.lang.reflect.InvocationHandler接口.
通過InvocationHandler接口實(shí)現(xiàn)的動(dòng)態(tài)代理只能代理接口的實(shí)現(xiàn)類.
動(dòng)態(tài)代理實(shí)現(xiàn)
1) 處理者(Handler)
public class DynaProxyHandler implements InvocationHandler { private Logger logger = Logger.getLogger(DynaProxyHandler.class); private Object target; //被代理對(duì)象 public void setTarget(Object target) { this.target = target; } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { logger.info("執(zhí)行開始時(shí)間:" + new Date()); Object result = method.invoke(target, args); logger.info("執(zhí)行結(jié)束時(shí)間:" + new Date()); return result; //返回method執(zhí)行結(jié)果 } }
2) 生產(chǎn)代理對(duì)象的工廠
import java.lang.reflect.Proxy; public class DynaProxyFactory { //obj為被代理對(duì)象 public static Object getProxy(Object obj){ DynaProxyHandler handler = new DynaProxyHandler(); handler.setTarget(obj); return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), handler); } }
3) 測(cè)試類
public class PersonTest { public static void main(String[] args) { IPerson person = (IPerson) DynaProxyFactory.getProxy(new Person()); //返回代理類,代理類是JVM在內(nèi)存中動(dòng)態(tài)創(chuàng)建的,該類實(shí)現(xiàn)傳入的接口數(shù)組的全部接口(的全部方法). person.eating(); person.sleep(); } }
總結(jié)
以上就是本文關(guān)于Spring靜態(tài)代理和動(dòng)態(tài)代理代碼詳解的全部?jī)?nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:
Spring常用配置及解析類說明
SpringMVC攔截器實(shí)現(xiàn)單點(diǎn)登錄
Java編程實(shí)現(xiàn)springMVC簡(jiǎn)單登錄實(shí)例
如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持!
文章名稱:Spring靜態(tài)代理和動(dòng)態(tài)代理代碼詳解
當(dāng)前路徑:http://aaarwkj.com/article8/iijiip.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站營(yíng)銷、搜索引擎優(yōu)化、全網(wǎng)營(yíng)銷推廣、動(dòng)態(tài)網(wǎng)站、定制開發(fā)、網(wǎng)站維護(hù)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)