前言
springboot团队之所以默认的代理模式设置成cglib代理,是因为他们认为使用cglib更不容易出现转换错误
This was changed in 1.4 (see https://github.com/spring-projects/spring-boot/issues/5423). We’ve generally found cglib proxies less likely to cause unexpected cast exceptions.
整体逻辑
先梳理一下整体的逻辑
jdk动态代理demo
interface IUserService{
void getUser();
}
class UserServiceImpl implements IUserService{
@Override
public void getUser() {
System.out.println("usersevice的getUser方法");
}
}
通过上面的图,在bean后置处理器的时候拿到当前的bean对象然后进行代理,最终返回代理对象;
//创建jdk代理的方法
/**
* 第一种 jDk
*/
public static Object createJdkProxy(Object currentObj){
return Proxy.newProxyInstance(SpringProxyDemo.class.getClassLoader(),currentObj.getClass().getInterfaces(),new Invoke(currentObj));
}
handler 在方法前后需要增强的功能 ;
class Invoke implements InvocationHandler {
private Object o;//真实的对象
public Invoke(Object object){
this.o = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("方法前后打印日志:--------before");
Object invoke = method.invoke(o, args); //执行对象的方法
System.out.println("方法前后打印日志--------after");
return invoke;
}
}
模拟一下:
通过反射创建对象 : userService
aop增强打印日志: userService对象传入handler里面 ,该对象才是真正执行方法的对象,返回代理对象
进行方法调用
public static void main(String[] args) throws Exception{
//1.这里类似于spring,简写哈:Spring - BeanDefine 根据里面的class信息进行反射创建对象
Class<UserServiceImpl> userServiceClass = UserServiceImpl.class;
UserServiceImpl userService = userServiceClass.newInstance();
//这里相当于spring的bean对象,现在要在service方法前后增加日志,所以要用aop, 调用bean的后置处理器
//创建代理对象 - 并返回,此时bean工厂里面的实例就是该代理对象
IUserService userServiceProxy = (IUserService)createJdkProxy(userService);
//调用方法
userServiceProxy.getUser();
}
看一下执行结果:
CGLBdemo
通过上面的图,在bean后置处理器的时候拿到当前的bean对象然后进行代理,最终返回代理对象;
class Inteceptor implements MethodInterceptor{
private Object obj;
public Inteceptor(Object object){
this.obj = object;
}
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("CGLB-方法前后打印日志:--------before");
Object invoke = methodProxy.invoke(obj, objects); //执行对象的方法
System.out.println("CGLB-方法前后打印日志--------after");
return invoke;
}
}
public static Object createCGLBProxy(Object currentObj){
Enhancer enhancer = new Enhancer();
// 设置enhancer对象的父类
enhancer.setSuperclass(currentObj.getClass());
// 设置enhancer的回调对象
enhancer.setCallback(new Inteceptor(currentObj));
// 创建代理对象
return enhancer.create();
}
public static void main(String[] args) throws Exception{
//1.这里类似于spring,简写哈:Spring - BeanDefine 根据里面的class信息进行反射创建对象
Class<UserServiceImpl> userServiceClass = UserServiceImpl.class;
UserServiceImpl userService = userServiceClass.newInstance();
//这里已经创建对象了,现在要在service方法前后增加日志,所以要用aop, 调用bean的后置处理器
//创建代理对象 - 并返回,此时bean工厂里面的实例就是该代理对象
UserServiceImpl cglbProxy = (UserServiceImpl)createCGLBProxy(userService);
cglbProxy.getUser();
}
JDK动态代理容易出现转换错误
原因: jdk动态生成的代理类, 是实现了接口,所以返回的类型是我们实现类的接口类型,如果我们在日常使用spring中,使用的是实现类的类型的话就会报错,如下:
看下面的例子可以看到,我使用的是jdk代理,custDeadLineWarnService创建了代理对象,返回的是
custDeadLineWarnService实现的接口类型
所以我们如果直接写CustDeadLineWarnServiceImpl就会报错:该类型不能转为接口类型
CGLB:
而 CGLIB 就不存在这个问题。因为 CGLIB 是通过继承生成子类来实现的,代理对象无论是赋值给接口还是实现类这两者都是代理对象的父类
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/105975.html