代理模式
代理(Proxy)是一種設(shè)計(jì)模式,提供了對(duì)目標(biāo)對(duì)象另外的訪問方式;即通過代理對(duì)象訪問目標(biāo)對(duì)象.這樣做的好處是:可以在目標(biāo)對(duì)象實(shí)現(xiàn)的基礎(chǔ)上,增強(qiáng)額外的功能操作,即擴(kuò)展目標(biāo)對(duì)象的功能。
這里使用到編程中的一個(gè)思想:不要隨意去修改別人已經(jīng)寫好的代碼或者方法,如果需改修改,可以通過代理的方式來擴(kuò)展該方法。
舉個(gè)例子來說明代理的作用:假設(shè)我們想邀請(qǐng)一位明星,那么并不是直接連接明星,而是聯(lián)系明星的經(jīng)紀(jì)人,來達(dá)到同樣的目的.明星就是一個(gè)目標(biāo)對(duì)象,他只要負(fù)責(zé)活動(dòng)中的節(jié)目,而其他瑣碎的事情就交給他的代理人(經(jīng)紀(jì)人)來解決.這就是代理思想在現(xiàn)實(shí)中的一個(gè)例子。
代理模式的 關(guān)鍵點(diǎn)是:代理對(duì)象與目標(biāo)對(duì)象.代理對(duì)象是對(duì)目標(biāo)對(duì)象的擴(kuò)展,并會(huì)調(diào)用目標(biāo)對(duì)象 .
靜態(tài)代理
靜態(tài)代理在使用時(shí),需要定義接口或者父類,被代理對(duì)象與代理對(duì)象一起實(shí)現(xiàn)相同的接口或者是繼承相同父類。
實(shí)例說明:
模擬保存動(dòng)作,定義一個(gè)保存動(dòng)作的接口:IUserDao.java,然后目標(biāo)對(duì)象實(shí)現(xiàn)這個(gè)接口的方法UserDao.java,此時(shí)如果使用靜態(tài)代理方式,就需要在代理對(duì)象(UserDaoProxy.java)中也實(shí)現(xiàn)IUserDao接口.調(diào)用的時(shí)候通過調(diào)用代理對(duì)象的方法來調(diào)用目標(biāo)對(duì)象。
需要 注意 的是,代理對(duì)象與目標(biāo)對(duì)象要實(shí)現(xiàn)相同的接口,然后通過調(diào)用相同的方法來調(diào)用目標(biāo)對(duì)象的方法。
接口:IUserDao.java
package net.ydstudio.service;/** * @author Nick * @projectName javaLean * @package net.ydstudio.service * @createDate 2018/08/16 15:35 * @updateDate 2018/08/16 15:35 */public interface IUserDao { /** * 保存數(shù)據(jù)庫(kù) * @param: [] * @return: void */ void save();}
目標(biāo)對(duì)象:UserDao.java
package net.ydstudio.service.impl;import net.ydstudio.service.IUserDao;/** * @author Nick * @projectName javaLean * @package net.ydstudio.service.impl * @createDate 2018/08/16 15:36 * @updateDate 2018/08/16 15:36 */public class UserDao implements IUserDao { /** * 保存數(shù)據(jù)庫(kù) * * @param: [] * @return: void */ public void save() { System.out.println("數(shù)據(jù)已經(jīng)保存到數(shù)據(jù)庫(kù)"); }}
代理對(duì)象:UserDaoProxy.java
package net.ydstudio.staticproxy;import net.ydstudio.service.IUserDao;import net.ydstudio.service.impl.UserDao;/** * @author Nick * @projectName javaLean * @package net.ydstudio.staticproxy * @createDate 2018/08/16 15:37 * @updateDate 2018/08/16 15:37 */public class UserDaoProxy implements IUserDao { /** * 保存被代理的對(duì)象 */ private UserDao target; public UserDaoProxy(UserDao target) { this.target = target; } /** * 保存數(shù)據(jù)庫(kù) * * @param: [] * @return: void */ public void save() { System.out.println("開始保存數(shù)據(jù)……"); target.save(); System.out.println("結(jié)束保存數(shù)據(jù)……"); }}
測(cè)試類:
package net.ydstudio.proxy;import net.ydstudio.service.IUserDao;import net.ydstudio.service.impl.UserDao;import org.junit.Test;import org.junit.runner.RunWith;import org.junit.runners.JUnit4;import static org.junit.Assert.*;/** * @author Nick * @projectName javaLean * @package net.ydstudio.proxy * @createDate 2018/08/16 15:58 * @updateDate 2018/08/16 15:58 */@RunWith(JUnit4.class)public class ProxyFactoryTest { @Test public void test(){ // 目標(biāo)對(duì)象 IUserDao target = new UserDao(); System.out.println(target.getClass()); // 給目標(biāo)對(duì)象,創(chuàng)建代理對(duì)象 IUserDao proxy = (IUserDao)new ProxyFactory(target).getProxyInstance(); // class $Proxy()內(nèi)存中動(dòng)態(tài)生成的代理對(duì)象 System.out.println(proxy.getClass()); // 執(zhí)行方法 代理對(duì)象 proxy.save(); }}
靜態(tài)代理總結(jié):
要解決上面靜態(tài)代理的缺點(diǎn),就必須使用動(dòng)態(tài)代理的方式。
動(dòng)態(tài)代理
動(dòng)態(tài)代理有以下特點(diǎn):
JDK中生成代理對(duì)象的api
JDK實(shí)現(xiàn)代理只需要使用靜態(tài)的newProxyInstance方法,該方法需要接收三個(gè)參數(shù):
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException
參數(shù)按順序解釋如下:
代碼實(shí)現(xiàn):
接口類IUserDao.java以及接口實(shí)現(xiàn)類,目標(biāo)對(duì)象UserDao是一樣的,沒有做修改.在這個(gè)基礎(chǔ)上,增加一個(gè)代理工廠類(ProxyFactory.java),將代理類寫在這個(gè)地方,然后在測(cè)試類(需要使用到代理的代碼)中先建立目標(biāo)對(duì)象和代理對(duì)象的聯(lián)系,然后代用代理對(duì)象的中同名方法。
代理工廠類ProxyFactory:
package net.ydstudio.proxy;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;/** * @author Nick * @projectName javaLean * @package net.ydstudio.proxy * @createDate 2018/08/16 15:44 * @updateDate 2018/08/16 15:44 */public class ProxyFactory { /** * 維護(hù)一個(gè)代理的目標(biāo)對(duì)象 */ private Object target; public ProxyFactory(Object target){ this.target = target; } /** * 給目標(biāo)對(duì)象生成代理對(duì)象 * @param: [] * @return: java.lang.Object */ public Object getProxyInstance(){ return Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(), new InvocationHandler() { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("開始事務(wù)2"); //執(zhí)行目標(biāo)對(duì)象方法 Object returnValue = method.invoke(target, args); System.out.println("提交事務(wù)2"); return returnValue; } } ); }}
測(cè)試類:
package net.ydstudio.proxy;import net.ydstudio.service.IUserDao;import net.ydstudio.service.impl.UserDao;import org.junit.Test;import org.junit.runner.RunWith;import org.junit.runners.JUnit4;import static org.junit.Assert.*;/** * @author Nick * @projectName javaLean * @package net.ydstudio.proxy * @createDate 2018/08/16 15:58 * @updateDate 2018/08/16 15:58 */@RunWith(JUnit4.class)public class ProxyFactoryTest { @Test public void test(){ // 目標(biāo)對(duì)象 IUserDao target = new UserDao(); System.out.println(target.getClass()); // 給目標(biāo)對(duì)象,創(chuàng)建代理對(duì)象 IUserDao proxy = (IUserDao)new ProxyFactory(target).getProxyInstance(); // class $Proxy()內(nèi)存中動(dòng)態(tài)生成的代理對(duì)象 System.out.println(proxy.getClass()); // 執(zhí)行方法 代理對(duì)象 proxy.save(); }}
JDK實(shí)現(xiàn)代理總結(jié):代理對(duì)象不需要實(shí)現(xiàn)接口,但是目標(biāo)對(duì)象一定要實(shí)現(xiàn)接口,否則不能用動(dòng)態(tài)代理。
總結(jié)
以上所述是小編給大家介紹的Java 中的三種代理模式,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)VeVb武林網(wǎng)網(wǎng)站的支持!
新聞熱點(diǎn)
疑難解答
圖片精選