1、JDK静态代理
- 业务接口
- 接口的实现类
- 代理类,实现接口,并扩展实现类的功能
2、JDK动态代理
- 业务接口
- 实现了业务接口的业务类
- 实现了InvocationHandler接口的handler代理类
3、CGLIB动态代理
不需要定义业务接口
- 业务类(不需要实现接口)
- cglib代理类(实现接口MethodInterceptor)
4、原理
1、JDK静态代理
一般是代理类对象直接包装被代理对象
2、JDK动态代理
- 是接口代理;
- 被代理类需要实现业务接口;
- 业务代理类需要实现InvocationHandler接口;
- JDK动态代理会根据被代理对象生成一个继承Proxy类,并实现该业务接口的JDK代理类,该类的字节码会被传进去的ClassLoader加载,创建JDK代理对象实例;
- JDK代理对象实例创建
- 创建业务代理对象实例;
- 1中的实例赋值给Proxy类;
- 通过2,JDK代理对象实例也就有了业务代理对象实例;
- JDK代理对象实例通过反射,根据被代理类的业务方法创建相应的Method对象(可能有多个);
- JDK代理对象实例调用业务方法时,会先把Method对象作为参数传递给invoke()方法(invoke方法的第二个参数);
- 调用JDK代理对象实例的invoke()回调方法,在invoke方法里通过反射来调用代理对象的方法;
3、CGLIB动态代理
- 是继承代理;
- 因为是继承,所有类和方法都不能用final修饰符;
5、JDK动态代理和CGLIB动态代理
JDK | CGLIB |
---|---|
只能代理实现了接口的类;没有实现接口的类不能实现JDK动态代理 | 针对类来实现代理;对指定目标类产生一个子类,通过方法拦截技术拦截所有父类方法的调用 |
6、三种代理方式的优缺点
JDK静态代理
缺点
- 只能为一个被代理类服务,如果需要代理多个类,会产生过多的代理类;
优点
- 在编译时产生class文件,运行时无需产生,可以直接使用,效率好;
JDK动态代理
缺点
- 必须实现接口;
- 通过反射动态代理方法,消耗系统性能;
优点
- 无需产生过多的代理类,避免重复代码,系统更加灵活
CGLIB动态代理
缺点
- CGLIB会继承被代理类;
- CGLIB会重写被代理方法;
- 代理类和代理方法不能使用final修饰符;
优点
- 无需实现接口,通过子类字节码来实现;
- 比反射速度快,没有性能问题;