• Java基础——动态代理


    1.什么是动态代理?

    简单的来说,就是本来让我自己做的事,请给别人来做,这个请的人就是代理对象

    那么动态代理就是在程序运行过程中产生这个代理对象,而程序运行中产生的对象就是用反射的来生成一个代理。

    举一个例子:

    有一个user对象,这个对象有四个方法分别是增,删,改,查。但是外界是不能直接调用这几个方法。你最多能执行查的操作,增、删、改的操作是不能执行的,你必须要加一个权限操作,应该看看你是否有权限执行这个操作。同理,谁操作了这个东西,你需要给我留下记录,免得我不知道是谁做的。所以,我应该在每一个方法的前面加权限校验,在每一个方法的后面加日志记录。

    该怎么做呢?

    有人说,很简单,直接在user对象的实现类里面去改,在增、删、改查前面加上权限校验,在后面加上日志记录。你能随便改别人的代码吗?你一改,所以用过user对象的地方都要改,这不乱套了吗?

    有人说,可以再重新创建一个user对象,在新对象中加上权限校验和日志记录。确实是这样。但是如果我还有一个学生类,还有一个老师类...等等,你每一个都新创建一个对象的话,太麻烦了,而且没有必要,因为对我来说,我只关心对象的增、删、改、查操作,对于权限校验和日志记录我并不关心,这个时候,我们可以找中介来做权限校验和日志记录的事情,这个中介就是动态代理对象!

    在java.lang.reflect包下提供一个Proxy类和一个InvocationHandle接口,通过这个类和接口可以实现生成动态代理对象。JDK提供的代理只能针对接口,而cglib则更加强大。

    Proxy类的方法创建动态代理对象

    Public static Object newProxyInstance(ClassLoader loader, Class<?> interfaces, InvocationHandle h);

    最终会调用InvocationHandle的方法invoke

    Object invoke(Object proxy, Method method, Object[] args);

    2.动态代理的实现

    下面用代码实现一下:

    创建一个接口,其中有四个方法

    /**
     * Created by YuKai Fan on 2018/9/25.
     */
    public interface UserDao {
        public abstract void add();
        public abstract void delete();
        public abstract void update();
        public abstract void find();
    }

    在创建该接口的实现类

    /**
     * Created by YuKai Fan on 2018/9/25.
     */
    public class UserDaoImpl implements UserDao {
        @Override
        public void add() {
            System.out.println("添加");
        }
    
        @Override
        public void delete() {
            System.out.println("删除");
        }
    
        @Override
        public void update() {
            System.out.println("修改");
        }
    
        @Override
        public void find() {
            System.out.println("查找");
        }
    }

    在创建一个类实现InvocationHandle接口

    /**
     * 该类实现了InvocationHandle接口
     * Created by YuKai Fan on 2018/9/25.
     */
    public class MyInvocationHandle implements InvocationHandler {
       private Object target; //目标对象
    
        public MyInvocationHandle(Object target) {
            this.target = target;
        }
    
        //重写invoke()方法
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("权限校验");
            Object invoke = method.invoke(target, args);
            System.out.println("日志记录");
            return invoke;//返回的就是代理对象
        }
    }

    创建测试类

    /**
     * Created by YuKai Fan on 2018/9/25.
     */
    public class Test {
        public static void main(String[] args) {
            UserDao userDao = new UserDaoImpl();
            userDao.add();
            userDao.delete();
            userDao.update();
            userDao.find();
            System.out.println("----------");
            //我们要创建一个动态代理对象,在Proxy类中有一个方法可以创建动态代理对象
            //public static Object newProxyInstance(ClassLoader loader, Class<?> interfaces, InvocationHandle h);
            //将userDao对象作为一个代理对象
            MyInvocationHandle myInvocationHandle = new MyInvocationHandle(userDao);
            UserDao proxy = (UserDao) Proxy.newProxyInstance(userDao.getClass().getClassLoader(), userDao.getClass().getInterfaces(), myInvocationHandle);
            proxy.find();
            proxy.add();
            proxy.update();
            proxy.delete();
        }
    }

    输出结果为

    添加
    删除
    修改
    查找
    ----------
    权限校验
    查找
    日志记录
    权限校验
    添加
    日志记录
    权限校验
    修改
    日志记录
    权限校验
    删除
    日志记录

    以上为JDK动态代理,只能针对接口做代理。我们有更强大的代理cglib。

    调用Proxy类中的newProxyInstance(ClassLoader loader,Class[] interfaces,InvocationHandler h)方法可以创建一个动态代理对象,但是这个方法需要3个参数,前两个参数是固定的,但第三个参数比较麻烦,需要我们创建一个类MyInvocationHandler来实现InvocationHandler接口,这个类里面要重写invoke()方法。

    JDK动态代理和cglib动态代理有什么区别? JDK动态代理智能对实现了接口的类生成代理对象; cglib可以对任意类生成代理对象,它的原理是对目标对象进行继承代理,如果目标对象被final修饰,那么该类无法被cglib代理。

    3.动态代理的应用

    Spring框架的一大特点就是AOP,SpringAOP的本质就是动态代理,而且Spring使用的是JDK与CGlib的混合代理。如果被代理的对象实现了接口,就优先使用jdk代理,如果没有实现接口,就是用cglib代理。

    AOP(Aspect-OrientedProgramming,面向切面编程),AOP包括切面(Aspect),通知(Advice),连接点(joinpoint),实现方式就是通过目标对象的代理在连接点前后加入通知,完成统一的切面操作。

    实现AOP的技术,主要分为两大类:

    一是动态代理,利用截取消息的方式。对消息进行装饰,以取代原有对象的执行。

    二是静态织入,引入特定的语法创建“方面”,从而是的编译器可以在编译期间织入有关“方面”的代码。

    Spring提供了两种方式来生成代理对象:JDKProxy和Cglib,具体使用哪种方式生成由AopProxyFactory根据AdviseSupport对象的配置来决定

    默认的策略是如果目标类是接口,则使用JDK动态代理技术,如果目标对象没有实现接口,则默认会采用CGLIB代理。

    如果目标对象实现了接口,可以强制使用CGLIB实现代理(添加CGLIB库,并在Spring配置中加入<aop:aspectj-autoproxy proxy-target-class="true" />)

    4.总结

    JDK动态代理

    1.因为利用JDKProxy生成的代理类实现了接口,所以目标类中所有的方法在代理类中都有。

    2.生成的代理类的所有方法都拦截了目标类的所有方法。而拦截器中invoke方法的内容正好就是代理类的各个方法的组成体。

    3.利用JDKProxy方法必须有接口的存在

    4.invoke方法中的三个参数可以访问目标类的被调用方法的API,别调用方法的参数,被调用方法的返回类型。

    cglib动态代理

    1.CGlib是一个强大的,高性能,高质量的code生成类库。他可以来运行气扩展Java类与实现Java接口

    2.用CGlib生成代理类是目标类的子类

    3.用CGlib生成代理类不需要接口

    4.用CGlib生成的代理类重写了父类的各个方法

    5.拦截器中的intercept方法内容正好就是代理类中的方法体

    Spring的两种代理方式

    1.若目标对象实现了若干接口,spring使用JDK的java.lang.reflect.Proxy类代理。

    优点:因为有接口,所以使系统更加松耦合

    缺点:为每个目标类创建接口

    2.若目标对象没有实现任何接口,spring使用cglib库生成目标对象的子类。

    优点:因为代理类与目标类是继承关系,所以不需要有接口的存在

    缺点:因为没有使用接口,所以系统的耦合性没有使用JDK的动态代理好。

  • 相关阅读:
    旧题复习{6}
    CF219D. Choosing Capital for Treeland [树形DP]
    POJ1947 Rebuilding Roads[树形背包]

    洛谷P1280 尼克的任务[DP]
    NOIP2003pj栈[卡特兰数]
    NOIP2001统计单词个数[序列DP]
    洛谷P1415 拆分数列[序列DP 状态 打印]
    POJ2828 Buy Tickets[树状数组第k小值 倒序]
    CF380C. Sereja and Brackets[线段树 区间合并]
  • 原文地址:https://www.cnblogs.com/FanJava/p/9700561.html
Copyright © 2020-2023  润新知