您好,登錄后才能下訂單哦!
本篇內容介紹了“Java代理模式怎么理解”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!
由于某些原因需要給某對象提供一個代理以控制對該對象的訪問。這時,訪問對象不適合或者不能直接引用目標對象,代理對象作為訪問對象和目標對象之間的中介。
比如在有些情況下,一個客戶不能或者不想直接訪問另一個對象,這時需要找一個中介幫忙完成某項任務,這個中介就是代理對象。例如,購買火車票不一定要去火車站買,可以通過 12306 網站或者去火車票代售點買。又如找女朋友、找保姆、找工作等都可以通過找中介完成。
靜態代理:由程序員創建代理類或特定工具自動生成源代碼再對其編譯,在程序運行前代理類的 .class 文件就已經存在了。
代碼實例:實現增刪改查操作,通過代理
接口:
package com.proxyPattern.staticProxy2; /** * @author wang * @version 1.0 * @packageName com.proxyPattern.staticProxy2 * @className UserService * @date 2021/12/27 17:54 * @Description 服務接口 */ public interface UserService { void add(); void delete(); void update(); void query(); }
真實類(這里是服務類)
package com.proxyPattern.staticProxy2; /** * @author wang * @version 1.0 * @packageName com.proxyPattern.staticProxy2 * @className UserServiceImp * @date 2021/12/27 17:55 * @Description 服務實現類 */ public class UserServiceImp implements UserService{ @Override public void add() { System.out.println("添加了一條數據"); } @Override public void delete() { System.out.println("刪除了一條數據"); } @Override public void update() { System.out.println("修改了一條數據"); } @Override public void query() { System.out.println("查詢了一條數據"); } }
代理類
package com.proxyPattern.staticProxy2; /** * @author wang * @version 1.0 * @packageName com.proxyPattern.staticProxy2 * @className UserServiceProxy * @date 2021/12/27 17:56 * @Description 服務代理類 */ public class UserServiceProxy implements UserService { private UserServiceImp userServiceImp; public UserServiceProxy() { } public void setUserServiceImp(UserServiceImp userServiceImp) { this.userServiceImp = userServiceImp; } @Override public void add() { getLog("add"); userServiceImp.add(); } @Override public void delete() { getLog("delete"); userServiceImp.delete(); } @Override public void update() { getLog("update"); userServiceImp.update(); } @Override public void query() { getLog("add"); userServiceImp.query(); } public void getLog(String message) { System.out.println("日志:" + message + "語句執行了"); } }
客戶端測試類
package com.proxyPattern.staticProxy2; /** * @author wang * @version 1.0 * @packageName com.proxyPattern.staticProxy2 * @className Customer * @date 2021/12/27 18:00 * @Description 客戶終端測試類 */ public class Customer { public static void main(String[] args) { UserServiceImp userServiceImp = new UserServiceImp(); UserServiceProxy p = new UserServiceProxy(); p.setUserServiceImp(userServiceImp); p.add(); p.update(); p.delete(); p.query(); } } /** * 執行結果: * 日志:add語句執行了 * 添加了一條數據 * 日志:update語句執行了 * 修改了一條數據 * 日志:delete語句執行了 * 刪除了一條數據 * 日志:add語句執行了 * 查詢了一條數據 */
上述代碼看到我們并沒有使用userServiceImp去執行方法,而是使用了一個代理類去執行,這就是代理模式,類似于你租房并沒有找房東租房,而是找的一個中間代理人中介來完成租房這個動作。
那么代理模式有哪些優點呢?
1、可以使得我們的真實角色更加純粹 ,不再去關注一些公共的事情
2、公共的業務由代理來完成 . 實現了業務的分工
3、公共業務發生擴展時變得更加集中和方便
缺點
1、代理模式會造成系統設計中類的數量增加
2、在客戶端和目標對象之間增加一個代理對象,會造成請求處理速度變慢;
3、增加了系統的復雜度;
如何解決這些問題呢?就靠下面的動態代理模式來解決
動態,是指在程序運行時,運用反射機制動態創建而成
沒錯,動態的代理模式使用的是反射,而且要自己寫一個動態代理類去動態的獲取一個代理類
代碼實例:案例同上,只不過采用的是動態代理模式
服務實現類(真實類)
package com.proxyPattern.staticProxy2; /** * @author wang * @version 1.0 * @packageName com.proxyPattern.staticProxy2 * @className Customer * @date 2021/12/27 18:00 * @Description 客戶終端測試類 */ public class Customer { public static void main(String[] args) { UserServiceImp userServiceImp = new UserServiceImp(); UserServiceProxy p = new UserServiceProxy(); p.setUserServiceImp(userServiceImp); p.add(); p.update(); p.delete(); p.query(); } } /** * 執行結果: * 日志:add語句執行了 * 添加了一條數據 * 日志:update語句執行了 * 修改了一條數據 * 日志:delete語句執行了 * 刪除了一條數據 * 日志:add語句執行了 * 查詢了一條數據 */
接口:
package com.proxyPattern. autoProxy; /** * @author wang * @version 1.0 * @packageName com.proxyPattern.staticProxy2 * @className UserService * @date 2021/12/27 17:54 * @Description 服務接口 */ public interface UserService { void add(); void delete(); void update(); void query(); }
動態代理類,這個幾乎可以做一個工具類使用,因為格式固定
package com.proxyPattern.autoProxy; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; /** * @author wang * @version 1.0 * @packageName com.proxyPattern.autoProxy * @className ProxyInvocationHandler * @date 2021/12/27 19:33 * @Description 動態代理類 */ public class ProxyInvocationHandler implements InvocationHandler { //被代理的接口 private Object target; public void setTarget(Object target) { this.target = target; } /** * @Date 2021/12/27 19:36 * @Param * @Return Object * @MetodName getProxy * @Author wang * @Description 生成得到代理類 */ public Object getProxy() { return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this); } /** * @Date 2021/12/27 19:34 * @Param * @param proxy * @param method * @param args * @Return Object * @MetodName invoke * @Author wang * @Description 處理代理實例,并返回結果 */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { log(method.getName()); Object result = method.invoke(target, args); return result; } public void log(String message) { System.out.println("日志:" + message + "語句執行了" ); } }
客戶端測試類:
package com.proxyPattern.autoProxy; /** * @author wang * @version 1.0 * @packageName com.proxyPattern.staticProxy2 * @className Customer * @date 2021/12/27 18:00 * @Description 客戶終端測試類 */ public class Customer { public static void main(String[] args) { //真實角色 UserService userService = new UserServiceImp(); //代理角色 ProxyInvocationHandler pih = new ProxyInvocationHandler(); //動態設置代理的對象 pih.setTarget(userService); //動態生成代理類 UserService proxy = (UserService) pih.getProxy(); proxy.query(); proxy.update(); } } /** * 日志:query語句執行了 * 查詢了一條數據 * 日志:update語句執行了 * 修改了一條數據 */
可以看到我們這里可以更方便的去獲取代理類了,只需要將動態設置代理類那里的對象改一下,就可以去代理別的類。
“Java代理模式怎么理解”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。