Spring的两种代理JDK和CGLIB

动态代理代码
测试代码

1.原理

java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。

而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。

1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP
2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP
3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换

JDK动态代理和CGLIB字节码生成的区别?
1)JDK动态代理只能对实现了接口的类生成代理,而不能针对类

  • 实现InvocationHandler
  • 使用Proxy.newProxyInstance产生代理对象
  • 被代理的对象必须要实现接口

2)CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法

  • 因为是继承,所以该类或方法最好不要声明成final
  • CGLib 必须依赖于CGLib的类库

2.JDK动态代理实现

2.1 业务类——被代理的类

public interface UserManager {
    public void addUser(String id, String password);
    public void delUser(String id);
}
public class UserManagerImpl implements UserManager {

    public void addUser(String id, String password) {
        System.out.println(".: 调用了UserManagerImpl.addUser()方法! ");

    }

    public void delUser(String id) {
        System.out.println(".: 调用了UserManagerImpl.delUser()方法! ");

    }
}

2.2 JDKProxy实现

public class JDKProxy implements InvocationHandler {

    private Object targetObject;//需要代理的目标对象

    public Object newProxy(Object targetObject) {//将目标对象传入进行代理
        this.targetObject = targetObject;
        return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),
                targetObject.getClass().getInterfaces(), this);//返回代理对象
    }

    public Object invoke(Object proxy, Method method, Object[] args)//invoke方法
            throws Throwable {
        checkPopedom();//一般我们进行逻辑处理的函数比如这个地方是模拟检查权限
        Object ret = null;      // 设置方法的返回值
        ret  = method.invoke(targetObject, args);       //调用invoke方法,ret存储该方法的返回值
        return ret;
    }

    private void checkPopedom() {//模拟检查权限的例子
        System.out.println(".:检查权限  checkPopedom()!");
    }
}

测试:

        JDKProxy jdkPrpxy = new JDKProxy();
        UserManager userManagerJDK = (UserManager) jdkPrpxy.newProxy(new UserManagerImpl());
        userManagerJDK.addUser("tom", "root");

结果:

.:检查权限  checkPopedom()!
.: 调用了UserManagerImpl.addUser()方法! 

2.3 CGLIB实现动态代理

public class CGlibProxy implements MethodInterceptor {

    private Object targetObject;// CGLib需要代理的目标对象

    public Object createProxyObject(Object obj) {
        this.targetObject = obj;
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(obj.getClass());
        enhancer.setCallback(this);
        Object proxyObj = enhancer.create();
        return proxyObj;// 返回代理对象
    }

    public Object intercept(Object proxy, Method method, Object[] args,
                            MethodProxy methodProxy) throws Throwable {
        Object obj = null;
        if ("addUser".equals(method.getName())) {// 过滤方法
            checkPopedom();// 检查权限
        }
        obj = method.invoke(targetObject, args);
        return obj;
    }

    private void checkPopedom() {
        System.out.println(".:检查权限  checkPopedom()!");
    }

}

测试:

       UserManager userManager = (UserManager) new CGlibProxy().createProxyObject(new UserManagerImpl());
        userManager.addUser("tom", "root");

结果:

.:检查权限  checkPopedom()!
.: 调用了UserManagerImpl.addUser()方法! 

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,198评论 19 139
  • 本文是我自己在秋招复习时的读书笔记,整理的知识点,也是为了防止忘记,尊重劳动成果,转载注明出处哦!如果你也喜欢,那...
    波波波先森阅读 12,343评论 6 86
  • 关于动态代理和静态代理 当一个对象(客户端)不能或者不想直接引用另一个对象(目标对象),这时可以应用代理模式在这两...
    carway阅读 4,951评论 2 26
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,993评论 6 342
  • 你说诗和远方 是天上的云朵 生活是幕后庄家 谎言像空气的密度 草坪装饰陷阱的躲藏 但我们还有翅膀 从爱的巢穴 追寻...
    没事虫虫茶阅读 355评论 0 2