您好,登錄后才能下訂單哦!
本篇文章為大家展示了Java中的代理模式有哪些,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。
代理模式
代理(Proxy)是一種設計模式,提供了對目標對象另外的訪問方式;即通過代理對象訪問目標對象.這樣做的好處是:可以在目標對象實現的基礎上,增強額外的功能操作,即擴展目標對象的功能。
這里使用到編程中的一個思想:不要隨意去修改別人已經寫好的代碼或者方法,如果需改修改,可以通過代理的方式來擴展該方法。
舉個例子來說明代理的作用:假設我們想邀請一位明星,那么并不是直接連接明星,而是聯系明星的經紀人,來達到同樣的目的.明星就是一個目標對象,他只要負責活動中的節目,而其他瑣碎的事情就交給他的代理人(經紀人)來解決.這就是代理思想在現實中的一個例子。
代理模式的 關鍵點是:代理對象與目標對象.代理對象是對目標對象的擴展,并會調用目標對象 .
靜態代理
靜態代理在使用時,需要定義接口或者父類,被代理對象與代理對象一起實現相同的接口或者是繼承相同父類。
實例說明:
模擬保存動作,定義一個保存動作的接口:IUserDao.java,然后目標對象實現這個接口的方法UserDao.java,此時如果使用靜態代理方式,就需要在代理對象(UserDaoProxy.java)中也實現IUserDao接口.調用的時候通過調用代理對象的方法來調用目標對象。
需要 注意 的是,代理對象與目標對象要實現相同的接口,然后通過調用相同的方法來調用目標對象的方法。
接口: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 { /** * 保存數據庫 * @param: [] * @return: void */ void save(); }
目標對象: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 { /** * 保存數據庫 * * @param: [] * @return: void */ public void save() { System.out.println("數據已經保存到數據庫"); } }
代理對象: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 { /** * 保存被代理的對象 */ private UserDao target; public UserDaoProxy(UserDao target) { this.target = target; } /** * 保存數據庫 * * @param: [] * @return: void */ public void save() { System.out.println("開始保存數據……"); target.save(); System.out.println("結束保存數據……"); } }
測試類:
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(){ // 目標對象 IUserDao target = new UserDao(); System.out.println(target.getClass()); // 給目標對象,創建代理對象 IUserDao proxy = (IUserDao)new ProxyFactory(target).getProxyInstance(); // class $Proxy()內存中動態生成的代理對象 System.out.println(proxy.getClass()); // 執行方法 代理對象 proxy.save(); } }
靜態代理總結:
可以在不修改代理目標對象的前提下,對代理目標的功能進行拓展。
需要實現代理目標對象實現的接口,一旦代理目標所實現的接口有修改,目標對象與代理都需要維護。
要解決上面靜態代理的缺點,就必須使用動態代理的方式。
動態代理
動態代理有以下特點:
代理對象,不需要實現接口
代理對象的生成,是利用JDK的API,動態的在內存中構建代理對象(需要我們指定創建代理對象/目標對象實現的接口的類型)
動態代理也叫做:JDK代理,接口代理
JDK中生成代理對象的api
JDK實現代理只需要使用靜態的newProxyInstance方法,該方法需要接收三個參數:
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException
參數按順序解釋如下:
ClassLoader loader,:指定當前目標對象使用類加載器,獲取加載器的方法是固定的
Class[] interfaces,:目標對象實現的接口的類型,使用泛型方式確認類型
InvocationHandler h:事件處理,執行目標對象的方法時,會觸發事件處理器的方法,會把當前執行目標對象的方法作為參數傳入
代碼實現:
接口類IUserDao.java以及接口實現類,目標對象UserDao是一樣的,沒有做修改.在這個基礎上,增加一個代理工廠類(ProxyFactory.java),將代理類寫在這個地方,然后在測試類(需要使用到代理的代碼)中先建立目標對象和代理對象的聯系,然后代用代理對象的中同名方法。
代理工廠類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 { /** * 維護一個代理的目標對象 */ private Object target; public ProxyFactory(Object target){ this.target = target; } /** * 給目標對象生成代理對象 * @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("開始事務2"); //執行目標對象方法 Object returnValue = method.invoke(target, args); System.out.println("提交事務2"); return returnValue; } } ); } }
測試類:
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(){ // 目標對象 IUserDao target = new UserDao(); System.out.println(target.getClass()); // 給目標對象,創建代理對象 IUserDao proxy = (IUserDao)new ProxyFactory(target).getProxyInstance(); // class $Proxy()內存中動態生成的代理對象 System.out.println(proxy.getClass()); // 執行方法 代理對象 proxy.save(); } }
JDK實現代理總結:代理對象不需要實現接口,但是目標對象一定要實現接口,否則不能用動態代理。
上述內容就是Java中的代理模式有哪些,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。