一、动态代理简介
优势:在不修改源码的情况下,对目标方法进行相应的增强。
作用:完成程序功能之间的松耦合。
二、动态代理的多种实现
- JDK代理:基于接口的动态代理技术(缺点,目标对象必须有接口,如果没有接口,则无法完成动态代理的实现)
- cglib代理:基于父类的动态代理技术
两者的区别如图所示:
1. 基于JDK的实现
目标接口类:
public interface TargetInterface { public void save(); public void print(String str); }
目标类:
public class Target implements TargetInterface{ public void save() { System.out.println("save running..."); } public void print(String str) { System.out.println(str); } }
增强类:
public class Advice { public void before() { System.out.println("前置增强"); } public void after() { System.out.println("后置增强"); } }
测试类:
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class ProxyTest { public static void main(String[] args) { //目标对象 final Target target = new Target(); //增强对象 final Advice advice = new Advice(); TargetInterface proxyInstance = (TargetInterface)Proxy.newProxyInstance( target.getClass().getClassLoader(), //目标对象类加载器 target.getClass().getInterfaces(), //目标对象相同的接口字节码对象数组 new InvocationHandler() { //调用代理对象的任何方法,实质执行的都是invoke方法 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{ advice.before(); //前置增强 Object invoke = method.invoke(target, args); //执行目标方法 advice.after(); //后置增强 System.out.println(); return invoke; } }); //代理对象的方法测试 proxyInstance.save(); proxyInstance.print("JDK动态代理"); } }
运行截图:
2. 基于cglib的实现
需要导入Jar包,如果是maven项目,则在pom.xml文件加入如下配置:
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>4.2.4.RELEASE</version> </dependency>
目标类:
public class Target { public void save() { System.out.println("save running..."); } public void print(String str) { System.out.println(str); } }
增强类:
public class Advice { public void before() { System.out.println("前置增强"); } public void after() { System.out.println("后置增强"); } }
测试类:
import java.lang.reflect.Method; import org.springframework.cglib.proxy.Enhancer; import org.springframework.cglib.proxy.MethodInterceptor; import org.springframework.cglib.proxy.MethodProxy; public class ProxyTest { public static void main(String[] args) { final Target target = new Target(); final Advice advice = new Advice(); //返回值就是动态生成的代理对象,基于cglib //创建增强器 Enhancer enhancer = new Enhancer(); //设置父类(目标) enhancer.setSuperclass(Target.class); //设置回调 enhancer.setCallback(new MethodInterceptor() { public Object intercept(Object o, Method method, Object[] obj, MethodProxy methodProxy) throws Throwable{ advice.before(); Object invoke = method.invoke(target, obj); advice.after(); System.out.println(); return invoke; } }); //创建代理对象 Target proxy = (Target)enhancer.create(); //测试代理方法 proxy.save(); proxy.print("基于cglib实现动态规划"); } }
运行截图:
三、为什么要有基于cglib的实现
使用JDK动态代理实现时,最大限制是被增强对象必须实现接口,并且增强的方法只能是接口中声明的方法。但在实际的项目中,可能总是存在对不实现业务接口的对象进行增强的需求,这时JDK动态代理将无能为力。
四、两种方式的适用场景
JDK动态代理
优点
- 不依赖第三方jar包, 使用方便
- 随着JDK的升级,JDK动态代理的性能在稳步提升
缺点
- 只能代理实现了接口的类
- 执行速度较慢
适用场景
- 如果你的程序需要频繁、反复地创建代理对象,则JDK动态代理在性能上更占优。
cglib
优点
由于是动态生成字节码实现代理,因此代理对象的执行速度较快, 约为JDK动态代理的1.5 ~ 2倍
可以代理没有实现接口的对象
缺点
- 不能代理final类
- 动态生成字节码虽然执行较快,但是生成速度很慢,根据网上一些人的测试结果,cglib创建代理对象的速度要比JDK慢10 ~ 15倍。
适用场景
- 不需要频繁创建代理对象的应用,如Spring中默认的单例bean,只需要在容器启动时生成一次代理对象。
以上就是Java 动态代理的多种实现方式的详细内容,更多关于Java 动态代理的实现的资料请关注自学编程网其它相关文章!
- 本文固定链接: https://zxbcw.cn/post/214293/
- 转载请注明:必须在正文中标注并保留原文链接
- QQ群: PHP高手阵营官方总群(344148542)
- QQ群: Yii2.0开发(304864863)