Spring面试之AOP(JDK动态代理实现)

Spring面试之AOP

什么是AOP

在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

对以上描述总结一下,在Spring中主要体现为以下几点:

  • Aspect Oriented Programming
  • AOP采取了横向抽取机制,取代了传统纵向继承体系重复性代码(性能监视、事务管理、安全检查、缓存)

解决问题方法

传统是这样的,每次权限校验前,进行调用接口的方法。
在这里插入图片描述

纵向继承

在没有AOP的时候,我们是这样解决权限校验的,写一个类,里面有权限校验的方法,然后子类继承这个类,子类里不仅有相关业务操作,同时因为继承了父类,所以同时有权限校验方法。可以对他进行直接调用。这样的话,只要需要权限校验就必须实现父类。

在这里插入图片描述

横向抽取机制

采用横向抽取机制,取代传统的继承。所谓横向抽取机制,就是一个代理机制,利用JDK动态代理产生一个代理类,在代理类中对相关业务方法进行一个代码增强。

  • Spring AOP使用纯Java实现,不需要专门的编译过程和类加载器,在运行期通过代理方式向目标类织入代码增强。

相关术语

  • JoinPoint(连接点):指那些被拦截到的点,在Spring中,这些点指的是方法,因为Spring只支持方法类型的连接点。
    • 所有可以被增强的方法,这些方法被称为连接点。
  • Pointcut(切入点):是指我们要对哪些Jointpoint进行拦截的定义。
    • 指向对某一个方法进行增强,被增强的方法称为切入点
  • Advice(通知/增强):是指拦截到JoinPoint之后要做的事情,通知分为:前置通知、后置通知、异常通知、最终通知、环绕通知。
    • 对某一个方法进行拦截后要做的事情
  • Introduction(引介):引介是一种特殊的通知在不修改类代码的前提下,Introduction可以在运行期为动态地添加一些方法或 Field(属性)。
    • 类增强技术,通过第三方来完成
  • Target(目标对象):代理的目标对象。
    • 被增强的对象,就是要增强的方法所属的那个类。
  • Weaving(织入):是指把增强应用到目标对象来创建新的代理对象的过程。Spring采用动态代理织入,而AspectJ采用编译期织入和类装载期织入。
    • 将Advice应用到Target的这个过程
  • Proxy(代理):一个类被AOP织入增强后,就产生一个结果代理类。
    • 织入过程中会产生代理类。
  • Aspect(切面):是切入点和通知(引介)的结合。
    • 方法和增强方法的组合

原理

JDK动态代理在这里插入图片描述

Proxy类中有一个静态方法,newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h),通过他可以生成一个代理对象,他需要传入三个参数:类加载器、类实现的接口、InvocationHandler接口,

对于InvocationHandler接口,可以通过匿名内部类来实现,也可以通过直接实现InvocationHandler接口。

InvocationHandler里有一个方法public Object invoke(Object proxy, Method method, Object[] args) throws Throwable,在调用目标类的任何一个方法的时候,都相当于执行了这个方法。所以我们只需要控制invoke这个方法就好了。

对于上面的第三个参数,传入接口的实例,所以直接传入this就好了,最后返回一个代理类。

    /**
     * 使用JDK动态代理,因为他可以对接口产生动态代理,创建UserDao的代理类
     * @return 代理对象
     */
    public Object createProxy(){
        Object proxy = Proxy.newProxyInstance(UserDao.class.getClassLoader(), UserDaoImpl.class.getInterfaces(), this);
        return proxy;
    }

对于invoke方法,我们可以通过method.getName()来判断需要对哪个方法(即切入点)来进行增强。

    /**
     * 在调用UserDao实现当中的增删查改都相当于调用invoke这个方法。
     * @return
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //如果是save方法的话,进行增强
        if ("save".equals(method.getName())){
            //todo  权限校验
            System.out.println("权限校验");
            return method.invoke(userDao, args);
        }
        return method.invoke(userDao, args);
    }

在这里插入图片描述

对于进行调用的时候,需要创建他的代理类来进行创建,

 public void demo2(){
        UserDao userDao = new UserDaoImpl();
        UserDao proxy = (UserDao) new MyJDKProxy(userDao).createProxy();
        proxy.save();
        proxy.find();
        proxy.update();
        proxy.delete();
    }

最后实现了对切入点的增强。

在这里插入图片描述

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值