Java 中的代理模式
代理程式是設計模式中的一種,它允許物件以另一個物件的形式出現,透過代理物件來存取原始物件。代理在許多應用程式中都被廣泛使用,其中最常見的應用是在網路中實現遠端物件呼叫和日誌記錄。
Java 中也有許多使用代理模式的範例。 Java 中的代理模式主要是透過以下三種方式實作:
# 的代理程式是透過在編譯階段建立代理類別來實現的。代理類別和原始類別實作了同樣的接口,代理類別透過呼叫原始類別的方法來實現其自身的方法。靜態代理的優點是比較簡單直觀,但缺點是需要手動編寫代理類,當需要代理的方法增加或修改時,需要對代理類進行相應的修改。
下面是一個簡單的靜態代理範例:
// 定义接口 interface UserManager { void add(); void delete(); } // 原始类 class UserManagerImpl implements UserManager { public void add() { System.out.println("添加用户"); } public void delete() { System.out.println("删除用户"); } } // 代理类 class UserManagerProxy implements UserManager { private UserManager userManager; public UserManagerProxy(UserManager userManager) { this.userManager = userManager; } public void add() { System.out.println("记录日志..."); userManager.add(); System.out.println("记录日志..."); } public void delete() { System.out.println("记录日志..."); userManager.delete(); System.out.println("记录日志..."); } } // 客户端代码 UserManager userManager = new UserManagerImpl(); UserManagerProxy userManagerProxy = new UserManagerProxy(userManager); userManagerProxy.add(); userManagerProxy.delete();
動態代理程式是在執行時間透過反射機制建立代理類別。和靜態代理不同,動態代理可以代理多個接口,並且不需要手動編寫代理類。動態代理的優點是比較靈活,但缺點是對效能有一定影響。
下面是一個簡單的動態代理範例:
// 定义接口 interface UserManager { void add(); void delete(); } // 原始类 class UserManagerImpl implements UserManager { public void add() { System.out.println("添加用户"); } public void delete() { System.out.println("删除用户"); } } // 代理类 class MyInvocationHandler implements InvocationHandler { private UserManager userManager; public MyInvocationHandler(UserManager userManager) { this.userManager = userManager; } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("记录日志..."); Object result = method.invoke(userManager, args); System.out.println("记录日志..."); return result; } } // 客户端代码 UserManager userManager = new UserManagerImpl(); InvocationHandler handler = new MyInvocationHandler(userManager); UserManager proxy = (UserManager) Proxy.newProxyInstance( userManager.getClass().getClassLoader(), userManager.getClass().getInterfaces(), handler ); proxy.add(); proxy.delete();
CGLIB 代理程式是透過產生原始類別的子類別來實現的。 CGLIB 代理可以代理沒有實作介面的類,而且比動態代理快。但是,CGLIB 代理也有缺點,它要求原始類別不能為 final 類型並且所有方法都不能為 final 類型。
下面是一個簡單的CGLIB 代理程式範例:
// 原始类 class UserManager { public void add() { System.out.println("添加用户"); } public void delete() { System.out.println("删除用户"); } } // 代理类 class MyMethodInterceptor implements MethodInterceptor { private UserManager userManager; public MyMethodInterceptor(UserManager userManager) { this.userManager = userManager; } public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { System.out.println("记录日志..."); Object result = method.invoke(userManager, args); System.out.println("记录日志..."); return result; } } // 客户端代码 UserManager userManager = new UserManager(); Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(UserManager.class); enhancer.setCallback(new MyMethodInterceptor(userManager)); UserManager proxy = (UserManager) enhancer.create(); proxy.add(); proxy.delete();
總結
#代理模式是常用的設計模式,它可以在不改變原始類別的情況下為其提供額外的功能。在 Java 中,代理模式有三種實作方式:靜態代理、動態代理和 CGLIB 代理。不同的實作方式各有優缺點,可以依具體情況選擇使用。
以上是Java 中的代理模式的詳細內容。更多資訊請關注PHP中文網其他相關文章!