本篇內容介紹了“Java動態代理如何實現”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!
JDK動態代理:利用反射機制生成一個實現代理接口的匿名類,在調用具體方法前調用InvokeHandler
來處理。
CGlib動態代理:利用ASM(開源的Java字節碼編輯庫,操作字節碼)開源包,將代理對象類的class文件加載進來,通過修改其字節碼生成子類來處理。
區別:JDK代理只能對實現接口的類生成代理;CGlib是針對類實現代理,對指定的類生成一個子類,并覆蓋其中的方法,這種通過繼承類的實現方式,不能代理final修飾的類。
強制使用CGlib
<!-- proxy-target-class="false"默認使用JDK動態代理 --> <aop:aspectj-autoproxy proxy-target-class="true"/> <aop-config proxy-target-class="true">
具體代碼示例:
/** * 目標接口類 */ public interface UserManager { public void addUser(String id, String password); public void delUser(String id); }
/** * 接口實現類 */ public class UserManagerImpl implements UserManager { @Override public void addUser(String id, String password) { System.out.println("調用了UserManagerImpl.addUser()方法!"); } @Override public void delUser(String id) { System.out.println("調用了UserManagerImpl.delUser()方法!"); } }
/** * JDK動態代理類 */ public class JDKProxy implements InvocationHandler { // 需要代理的目標對象 private Object targetObject; public Object newProxy(Object targetObject) { // 將目標對象傳入進行代理 this.targetObject = targetObject; // 返回代理對象 return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this); } // invoke方法 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // 進行邏輯處理的函數 checkPopedom(); Object ret = null; // 調用invoke方法 ret = method.invoke(targetObject, args); return ret; } private void checkPopedom() { // 模擬檢查權限 System.out.println("檢查權限:checkPopedom()!"); } }
/** * CGlib動態代理類 */ public class CGLibProxy implements MethodInterceptor { // CGlib需要代理的目標對象 private Object targetObject; public Object createProxyObject(Object obj) { this.targetObject = obj; Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(obj.getClass()); enhancer.setCallback(this); Object proxyObj = enhancer.create(); return proxyObj; } @Override public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { Object obj = null; // 過濾方法 if ("addUser".equals(method.getName())) { // 檢查權限 checkPopedom(); } obj = method.invoke(targetObject, args); return obj; } private void checkPopedom() { System.out.println("檢查權限:checkPopedom()!"); } }
/** * 測試類 */ public class ProxyTest { public static void main(String[] args) { UserManager userManager = (UserManager)new CGLibProxy().createProxyObject(new UserManagerImpl()); System.out.println("CGLibProxy:"); userManager.addUser("tom", "root"); System.out.println("JDKProxy:"); JDKProxy jdkProxy = new JDKProxy(); UserManager userManagerJDK = (UserManager)jdkProxy.newProxy(new UserManagerImpl()); userManagerJDK.addUser("tom", "root"); } }
// 運行結果
CGLibProxy:
檢查權限checkPopedom()!
調用了UserManagerImpl.addUser()方法!
JDKProxy:
檢查權限checkPopedom()!
掉用了UserManagerImpl.addUser()方法!
“Java動態代理如何實現”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。