什么是代理模式?
成都創(chuàng)新互聯(lián)公司自2013年起,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目成都網(wǎng)站設(shè)計(jì)、成都做網(wǎng)站網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢想脫穎而出為使命,1280元南明做網(wǎng)站,已為上家服務(wù),為南明各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:028-86922220
代理模式:在調(diào)用處不直接調(diào)用目標(biāo)類進(jìn)行操作,而是調(diào)用代理類,然后通過代理類來調(diào)用目標(biāo)類進(jìn)行操作。在代理類調(diào)用目標(biāo)類的前后可以添加一些預(yù)處理和后處理操作來完成一些不屬于目標(biāo)類的功能。
為什么要使用代理模式?
通過代理模式可以實(shí)現(xiàn)對目標(biāo)類調(diào)用的控制、在目標(biāo)類調(diào)用前/后進(jìn)行一些不屬于目標(biāo)類的操作,如:數(shù)據(jù)驗(yàn)證、預(yù)處理、后處理、異常處理等
什么是靜態(tài)代理什么是動態(tài)代理?
jdk代理和cglib代理區(qū)別在哪里?
不同點(diǎn):jdk動態(tài)代理的目標(biāo)類必須實(shí)現(xiàn)的有接口,因?yàn)樵谡{(diào)用Proxy.newProxyInstance()的時(shí)候需要傳入目標(biāo)類的接口類。而cglib不做此限制。
下面看代碼分析:
定義一個(gè)Person接口
package com.zpj.designMode.proxy; //定義一個(gè)Person接口 public interface Person { public void doWork(); }
添加一個(gè)實(shí)現(xiàn)類:MrLi
package com.zpj.designMode.proxy; //添加一個(gè)實(shí)現(xiàn)類 public class MrLi implements Person { @Override public void doWork() { System.out.println("-----doWork"); } }
靜態(tài)代理:
添加一個(gè)靜態(tài)代理類Proxy
package com.zpj.designMode.proxy; //靜態(tài)代理,代理必須和目標(biāo)類實(shí)現(xiàn)共同的接口 public class Proxy implements Person { private Person person;// 被代理人 //這里的目標(biāo)類型決定了該代理類只能代理實(shí)現(xiàn)了Person接口的實(shí)例,而不能接收其他類型參數(shù),這也就是靜態(tài)代理的局限性 public Proxy(Person person) { this.person = person; } @Override public void doWork() { System.out.println("doSomething-----start"); person.doWork(); System.out.println("doSomething-----end"); } }
靜態(tài)代理測試程序:
package com.zpj.designMode.proxy; public class Run { public static void main(String[] args) { MrLi li = new MrLi(); Proxy proxy = new Proxy(li); //調(diào)用處直接調(diào)用代理進(jìn)行目標(biāo)方法的操作。 proxy.doWork(); } }
JDK動態(tài)代理:
添加一個(gè)代理JDKProxy,該代理實(shí)現(xiàn)InvocationHandler接口且覆寫invoke方法。
package com.zpj.designMode.proxy.jdk; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; /*** @author Perkins Zhu @date 2017年3月13日 上午8:41:10 */ public class JDKProxy implements InvocationHandler { private Object person;// 被代理人 //這里的目標(biāo)類型為Object,則可以接受任意一種參數(shù)作為被代理類,實(shí)現(xiàn)了動態(tài)代理。但是要注意下面的newProxyInstance()中的參數(shù) public Object getInstance(Object person) { this.person = person; //與cglib的區(qū)別在于這里構(gòu)建代理對象的時(shí)候需要傳入被代理對象的接口對象,第二個(gè)參數(shù)。而cglib不需要被代理對象實(shí)現(xiàn)任何接口即可 return Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(), this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("doSomething---------start"); method.invoke(person, args); System.out.println("doSomething---------end"); return null; } }
JDK動態(tài)代理測試程序
package com.zpj.designMode.proxy.jdk; import com.zpj.designMode.proxy.MrLi; import com.zpj.designMode.proxy.Person; /*** * @author Perkins Zhu * @date 2017年3月13日 上午8:51:31 */ public class Run { public static void main(String[] args) { Person person = (Person) new JDKProxy().getInstance(new MrLi()); //注意這里的person不是目標(biāo)類person,而是代理類person:debug的時(shí)候顯示null,有'$'標(biāo)識符 person.doWork(); } }
Cglib動態(tài)代理:
添加一個(gè)CglibProxy代理,同時(shí)實(shí)現(xiàn)MethodInterceptor接口。
package com.zpj.designMode.proxy.cglib; import java.lang.reflect.Method; import net.sf.cglib.proxy.Enhancer; import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy; /*** * @author Perkins Zhu * @date 2017年3月13日 上午9:02:54 */ public class CglibProxy implements MethodInterceptor { private Object targetObject; // 這里的目標(biāo)類型為Object,則可以接受任意一種參數(shù)作為被代理類,實(shí)現(xiàn)了動態(tài)代理 public Object getInstance(Object target) { this.targetObject = target; Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(target.getClass()); enhancer.setCallback(this); //注意該處代理的創(chuàng)建過程 Object proxyObj = enhancer.create(); return proxyObj;// 返回代理對象 } @Override public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { Object obj = null; System.out.println("doSomething---------start"); obj = method.invoke(targetObject, args); System.out.println("doSomething---------end"); return obj; } }
Cglib動態(tài)代理測試程序
package com.zpj.designMode.proxy.cglib; import com.zpj.designMode.proxy.MrLi; import com.zpj.designMode.proxy.Person; /*** @author Perkins Zhu @date 2017年3月13日 上午9:07:38 */ public class Run { public static void main(String[] args) { Person person = (Person)new CglibProxy().getInstance(new MrLi()); person.doWork(); } }
仔細(xì)對比Proxy、CglibProxy和JDKProxy區(qū)分靜態(tài)代理、JDK動態(tài)代理和Cglib動態(tài)代理的異同點(diǎn)!
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。