<返回更多

java静态代理与动态代理

2019-12-09    
加入收藏

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);
}

运行结果如下:

 

java静态代理与动态代理

 

 

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);
 }
}

运行结果如下:

 

java静态代理与动态代理

 

 

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);
}

运行结果如下:

 

java静态代理与动态代理

 

 

4.总结

静态代理和动态代理都需要编写代理类,不同的是动态代理不需要批量编写代理类,只是需要编写一个通用代理类就行了。JDK动态代理中实现类必须实现接口,CGLIB则没有此限制。

代理的使用场景:鉴权、打印方法前后日志、减少代码的侵入性。

声明:本站部分内容来自互联网,如有版权侵犯或其他问题请与我们联系,我们将立即删除或处理。
▍相关推荐
更多资讯 >>>