1. 靜態代理與動態代理區別
靜態代理:需要編寫批量代理類
動態代理:不需要編寫批量代理類
2.靜態代理
靜態代理結構如下:
編寫測試接口:
public interface TestInteface { /** * 測試方法 */ String runNow(String param); }//end
編寫測試類:
public class TestProxy implements TestInteface { public String runNow(String param) { System.out.println("param:" + param); return "result:" + param; } }//end
編寫靜態代理類:
public class StaticProxy implements TestInteface { private TestInteface testInteface; public StaticProxy(TestInteface testInteface) { this.testInteface = testInteface; } @Override public String runNow(String param) { System.out.println("靜態代理開始"); String result = testInteface.runNow(param); System.out.println("靜態代理結束"); return result; } }//end
編寫main方法:
public static void main(String[] args) { TestInteface testInteface = new TestProxy(); StaticProxy staticProxy = new StaticProxy(testInteface); String s = staticProxy.runNow("這是靜態代理"); System.out.println(s); }
運行結果如下:
3.動態代理
動態代理分為JDK動態代理、CGLIB動態代理。
JDK動態代理結構如下:
編寫通用代理類:
public class JdkProxy implements InvocationHandler { /** * 目標類 */ private Object target; public JdkProxy(Object target) { this.target = target; } /* * @Param: proxy 動態代理的實例 * @Param: method 目標類的方法 * @Param: args 目標類方法的參數 * @Return: 目標類方法的執行結果 * @Description: */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("JDK動態代理開始"); Object result = method.invoke(target, args); System.out.println("JDK動態代理結束"); return result; } }//end
編寫main方法:
public static void main(String[] args) { try { TestInteface testInteface = new TestProxy(); JdkProxy jdkProxy = new JdkProxy(testInteface); TestInteface testProxy_ = (TestInteface) Proxy.newProxyInstance(testInteface.getClass().getClassLoader(), testInteface.getClass().getInterfaces(), jdkProxy); String r = testProxy_.runNow("這是JDK動態代理"); System.out.println(r); } catch (Exception e) { System.out.println(e); } }
運行結果如下:
CGLIB動態代理結構如下:
由于JDK動態代理是JDK原生,CGLIB動態代理是第三方,所以需要引入如下jar包:
compile 'cglib:cglib:3.3.0' compile group: 'org.Apache.ant', name: 'ant', version: '1.10.3' compile group: 'org.ow2.asm', name: 'asm', version: '7.1'
編寫通用代理類:
public class CGLIBProxy implements MethodInterceptor { /** * @Param: obj 目標對象 * @Param: method 目標對象方法 * @Param: args 目標對象方法參數 * @Param: proxy 代理類方法 * @Return: */ @Override public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { System.out.println("CGLIB動態代理開始"); Object result = proxy.invokeSuper(obj, args); System.out.println("CGLIB動態代理結束"); return result; } }//end
編寫main方法:
public static void main(String[] args) { CGLIBProxy cglibProxy = new CGLIBProxy(); //強化劑實例化 Enhancer enhancer = new Enhancer(); //強化目標類 enhancer.setSuperclass(TestProxy.class); //使用通用代理類進行強化 enhancer.setCallback(cglibProxy); //創建目標對象 TestProxy o = (TestProxy) enhancer.create(); String s = o.runNow("這是CGLIB動態代理"); System.out.println(s); }
運行結果如下:
4.總結
靜態代理和動態代理都需要編寫代理類,不同的是動態代理不需要批量編寫代理類,只是需要編寫一個通用代理類就行了。JDK動態代理中實現類必須實現接口,CGLIB則沒有此限制。
代理的使用場景:鑒權、打印方法前后日志、減少代碼的侵入性。