1. 静态代理与动态代理区别
静态代理:需要编写批量代理类
【java静态代理与动态代理】动态代理:不需要编写批量代理类
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则没有此限制 。
代理的使用场景:鉴权、打印方法前后日志、减少代码的侵入性 。
推荐阅读
- Javascript中的8种常见数据结构
- WordPress 伪静态规则设置:Apache和Nginx,以及二级目录规则
- 零基础学习Java之运算符
- Java中有哪些无锁技术来解决并发问题?如何使用?
- 史上最全 Java 中各种锁的介绍
- 想开个淘宝店怎么找货源 淘宝怎么做代理卖商品
- Java异常处理只有Try-Catch吗?
- APP测试过程中Proxifier代理方法
- JavaScript原型详解
- 教你怎么从Java8升级到Java11