Spring三

Spring三

AOP

什么是AOP?

  • AOP:全称是 Aspect Oriented Programming 即:面向切面编程
  • 简单的说它就是把我们程序重复的代码抽取出来,在需要执行的时候,使用动态代理的技术,在不修改源码的基础上,对我们的已有方法进行增强。就是不再需要创建一个实体类去给某些类的方法进行增强,只需要将这些增强的代码抽取出来为一个类,然后配置Spring,在使用需要增强方法的时候由Spring自己动态代理的给你增强。

    AOP 的作用及优势

  • 作用:
    • 在程序运行期间,不修改源码对已有方法进行增强。
  • 优势:
    1. 减少重复代码
    2. 提高开发效率
    3. 维护方便

      AOP 的实现方式

      使用动态代理技术。

动态代理

动态代理:

  • 特点:字节码随用随创建,随用随加载

  • 作用:不修改源码的基础上对方法增强

  • 类:

    1. 基于接口的动态代理
    2. 基于子类的动态代理
  • 基于接口的动态代理:

    • 涉及的类:Proxy
    • 提供者:JDK官方
    • 如何创建代理对象:
      • 使用Proxy类中的newProxyInstance方法
      • 创建代理对象的要求:
        • 被代理类最少实现一个接口,如果没有则不能使用
        • newProxyInstance方法的参数:
          1. ClassLoader:类加载器
            • 它是用于加载代理对象字节码的。和被代理对象使用相同的类加载器。固定写法。
          2. Class[]:字节码数组
            • 它是用于让代理对象和被代理对象有相同方法。固定写法。
          3. InvocationHandler:用于提供增强的代码
            • 它是让我们写如何代理。我们一般都是些一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的。
            • 此接口的实现类都是谁用谁写。
            • 实现此接口时会重写invoke方法,其中的参数proxy代表代理对象的引用,method代表当前执行的方法,args当前执行方法所需的参数,return返回值为和被代理对象方法有相同的返回值.
  • 基于子类的动态代理

    • 涉及的类:Enhancer
    • 提供者:第三方cglib库
    • 如何创建代理对象:
      • 使用Enhancer类中的create方法
      • 创建代理对象的要求:
        • 被代理类不能是最终类
      • create方法的参数:
        1. Class:字节码
          • 它是用于指定被代理对象的字节码。
        2. Callback:用于提供增强的代码
          • 它是让我们写如何代理。我们一般都是些一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的。
          • 此接口的实现类都是谁用谁写。
          • 我们一般写的都是该接口的子接口实现类:MethodInterceptor

AOP的相关术语

  1. Joinpoint(连接点):所谓连接点是指那些被拦截到的点。在spring中,这些点指的是方法,因为spring只支持方法类型的连接点。就是所有方法。
  2. Pointcut(切入点):所谓切入点是指我们要对哪些Joinpoint进行拦截的定义。就是需要动态代理增强的方法。
  3. Advice(通知/增强):所谓通知是指拦截到Joinpoint之后所要做的事情就是通知。就是对要增强的方法干什么。
    • 通知的类型:前置通知,后置通知,异常通知,最终通知,环绕通知。
      tzjbTK.md.jpg
  4. Introduction(引介):引介是一种特殊的通知在不修改类代码的前提下, Introduction可以在运行期为类动态地添加一些方法或Field。
  5. Target(目标对象):代理的目标对象。
  6. Weaving(织入):是指把增强应用到目标对象来创建新的代理对象的过程。spring 采用动态代理织入,而 AspectJ 采用编译期织入和类装载期织入。就是动态代理增强的这个过程。
  7. Proxy(代理):一个类被 AOP 织入增强后,就产生一个结果代理类。
  8. Aspect(切面):是切入点和通知(引介)的结合。

Spring中基于XML的AOP配置

  • 配置步骤

    1. 把通知Bean也交给spring来管理
    2. 使用aop:config标签表明开始AOP的配置
    3. 使用aop:aspect标签表明配置切面
      • id属性:是给切面提供一个唯一标识
      • ref属性:是指定通知类bean的Id。
    4. 在aop:aspect标签的内部使用对应标签来配置通知的类型
      • aop:before:表示配置前置通知
        • method属性:用于指定Logger类中哪个方法是前置通知
        • pointcut属性:用于指定切入点表达式,该表达式的含义指的是对业务层中哪些方法增强
  • 切入点表达式的写法:

    • 关键字:execution(表达式)
    • 表达式:
      • 访问修饰符 返回值 包名.包名.包名…类名.方法名(参数列表)
      • 标准的表达式写法:
        public void com.itheima.service.impl.AccountServiceImpl.saveAccount()
      • 访问修饰符可以省略
        void com.itheima.service.impl.AccountServiceImpl.saveAccount()
      • 返回值可以使用通配符,表示任意返回值
        * com.itheima.service.impl.AccountServiceImpl.saveAccount()
      • 包名可以使用通配符,表示任意包。但是有几级包,就需要写几个.
        `
        ....AccountServiceImpl.saveAccount())`
      • 包名可以使用..表示当前包及其子包
        *..AccountServiceImpl.saveAccount()
      • 类名和方法名都可以使用来实现通配
        `
        ...*()`
      • 参数列表:
        • 可以直接写数据类型:
          • 基本类型直接写名称 int
          • 引用类型写包名.类名的方式 java.lang.String
          • 可以使用通配符表示任意类型,但是必须有参数
          • 可以使用..表示有无参数均可,有参数可以是任意类型
      • 全通配写法:
        * *..*.*(..)
      • 实际开发中切入点表达式的通常写法:
        • 切到业务层实现类下的所有方法
          * com.itheima.service.impl.*.*(..)
  • Spring中基于注解配置环绕通知

  • 问题:

    • 当我们配置了环绕通知之后,切入点方法没有执行,而通知方法执行了。
  • 分析:

    • 通过对比动态代理中的环绕通知代码,发现动态代理的环绕通知有明确的切入点方法调用,而我们的代码中没有。
  • 解决:

    • Spring框架为我们提供了一个接口:ProceedingJoinPoint。该接口有一个方法proceed(),此方法就相当于明确调用切入点方法。
    • 该接口可以作为环绕通知的方法参数,在程序执行时,spring框架会为我们提供该接口的实现类供我们使用。
  • spring中的环绕通知:

    • 它是spring框架为我们提供的一种可以在代码中手动控制增强方法何时执行的方式。

Spring中基于注解的AOP配置

  • 首先需要导入关于注解及aop的约束。

  • 步骤:

    1. 在通知类上使用@Aspect注解声明为切面。

    2. 在增强的方法上使用注解配置通知

      1. @Before
        • 作用:把当前方法看成是前置通知。
        • 属性:value:用于指定切入点表达式,还可以指定切入点表达式的引用。
      2. @AfterReturning
        • 作用:把当前方法看成是后置通知。
        • 属性:value:用于指定切入点表达式,还可以指定切入点表达式的引用
      3. @AfterThrowing
        • 作用:把当前方法看成是异常通知。
        • 属性:value:用于指定切入点表达式,还可以指定切入点表达式的引用
      4. @After
        • 作用:把当前方法看成是最终通知。
        • 属性:value:用于指定切入点表达式,还可以指定切入点表达式的引用
      5. @Around
        • 作用:把当前方法看成是环绕通知。
        • 属性:value:用于指定切入点表达式,还可以指定切入点表达式的引用。
    3. 在配置文件中开启对注解AOP的支持

      <aop:aspectj-autoproxy/>
  • 切入点表达式注解,使用@Pointcut

    @Pointcut("execution(* com.itheima.service.impl.*.*(..))")
    private void pt1() {}
    应用时
    @Before("pt1()")//注意:千万别忘了写括号
打赏
  • 版权声明: 本博客所有文章除特别声明外,均采用 Apache License 2.0 许可协议。转载请注明出处!

请我喝杯咖啡吧~

支付宝
微信