兩種方式為:1、JDK動(dòng)態(tài)代理,利用反射機(jī)制生成一個(gè)實(shí)現(xiàn)代理接口的匿名類,在調(diào)用具體方法前調(diào)用InvokeHandler來(lái)處理;2、CGLIB動(dòng)態(tài)代理,利用asm開(kāi)源包,對(duì)代理對(duì)象類的class文件加載進(jìn)來(lái),通過(guò)修改其字節(jié)碼生成子類來(lái)處理。
本教程操作環(huán)境:windows7系統(tǒng)、java8版、DELL G3電腦。
動(dòng)態(tài)代理是反射的一個(gè)非常重要的應(yīng)用場(chǎng)景。動(dòng)態(tài)代理常被用于一些 Java 框架中。例如 Spring 的 AOP ,Dubbo 的 SPI 接口,就是基于 Java 動(dòng)態(tài)代理實(shí)現(xiàn)的。
動(dòng)態(tài)代理的方式有兩種:
-
JDK動(dòng)態(tài)代理:利用反射機(jī)制生成一個(gè)實(shí)現(xiàn)代理接口的匿名類,在調(diào)用具體方法前調(diào)用InvokeHandler來(lái)處理。
-
CGLIB動(dòng)態(tài)代理:利用ASM(開(kāi)源的Java字節(jié)碼編輯庫(kù),操作字節(jié)碼)開(kāi)源包,將代理對(duì)象類的class文件加載進(jìn)來(lái),通過(guò)修改其字節(jié)碼生成子類來(lái)處理。
區(qū)別:JDK代理只能對(duì)實(shí)現(xiàn)接口的類生成代理;CGlib是針對(duì)類實(shí)現(xiàn)代理,對(duì)指定的類生成一個(gè)子類,并覆蓋其中的方法,這種通過(guò)繼承類的實(shí)現(xiàn)方式,不能代理final修飾的類。
強(qiáng)制使用CGlib
<!-- proxy-target-class="false"默認(rèn)使用JDK動(dòng)態(tài)代理 --> <aop:aspectj-autoproxy proxy-target-class="true"/> <aop-config proxy-target-class="true"> <!-- 切面詳細(xì)配置 --> </aop-config>
具體代碼示例:
/** * 目標(biāo)接口類 */ public interface UserManager { public void addUser(String id, String password); public void delUser(String id); }
/** * 接口實(shí)現(xiàn)類 */ public class UserManagerImpl implements UserManager { @Override public void addUser(String id, String password) { System.out.println("調(diào)用了UserManagerImpl.addUser()方法!"); } @Override public void delUser(String id) { System.out.println("調(diào)用了UserManagerImpl.delUser()方法!"); } }
/** * JDK動(dòng)態(tài)代理類 */ public class JDKProxy implements InvocationHandler { // 需要代理的目標(biāo)對(duì)象 private Object targetObject; public Object newProxy(Object targetObject) { // 將目標(biāo)對(duì)象傳入進(jìn)行代理 this.targetObject = targetObject; // 返回代理對(duì)象 return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this); } // invoke方法 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // 進(jìn)行邏輯處理的函數(shù) checkPopedom(); Object ret = null; // 調(diào)用invoke方法 ret = method.invoke(targetObject, args); return ret; } private void checkPopedom() { // 模擬檢查權(quán)限 System.out.println("檢查權(quán)限:checkPopedom()!"); } }
/** * CGlib動(dòng)態(tài)代理類 */ public class CGLibProxy implements MethodInterceptor { // CGlib需要代理的目標(biāo)對(duì)象 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; // 過(guò)濾方法 if ("addUser".equals(method.getName())) { // 檢查權(quán)限 checkPopedom(); } obj = method.invoke(targetObject, args); return obj; } private void checkPopedom() { System.out.println("檢查權(quán)限:checkPopedom()!"); } }
/** * 測(cè)試類 */ 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"); } }
// 運(yùn)行結(jié)果 CGLibProxy: 檢查權(quán)限checkPopedom()! 調(diào)用了UserManagerImpl.addUser()方法! JDKProxy: 檢查權(quán)限checkPopedom()! 掉用了UserManagerImpl.addUser()方法!
總結(jié):
1、JDK代理使用的是反射機(jī)制實(shí)現(xiàn)aop的動(dòng)態(tài)代理,CGLIB代理使用字節(jié)碼處理框架asm,通過(guò)修改字節(jié)碼生成子類。所以jdk動(dòng)態(tài)代理的方式創(chuàng)建代理對(duì)象效率較高,執(zhí)行效率較低,cglib創(chuàng)建效率較低,執(zhí)行效率高;
2、JDK動(dòng)態(tài)代理機(jī)制是委托機(jī)制,具體說(shuō)動(dòng)態(tài)實(shí)現(xiàn)接口類,在動(dòng)態(tài)生成的實(shí)現(xiàn)類里面委托hanlder去調(diào)用原始實(shí)現(xiàn)類方法,CGLIB則使用的繼承機(jī)制,具體說(shuō)被代理類和代理類是繼承關(guān)系,所以代理類是可以賦值給被代理類的,如果被代理類有接口,那么代理類也可以賦值給接口。
(推薦教程:java入門(mén)教程)