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则没有此限制。
代理的使用场景:鉴权、打印方法前后日志、减少代码的侵入性。