Spring两种实现动态代理(JDK与CGLIB)的讲解与demo

前言

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 是通过继承生成子类来实现的,代理对象无论是赋值给接口还是实现类这两者都是代理对象的父类 

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

爱敲代码的小松

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值