• Spring AOP详解


    Spring AOP详解

    AOP概念

      AOP是面向切面(方面)编程,扩展功能不修改源代码实现,其采取横向抽取机制,取代了传统纵向继承体系重复性代码。在其底层,使用动态代理来实现,对于有接口情况,使用动态代理创建接口实现类代理对象;对于没有接口情况,使用动态代理创建类的子类代理对象。

    AOP操作术语  

      Joinpoint(连接点):类里面可以被增强的方法,这些方法称为连接点。
      Pointcut(切入点):所谓切入点是指我们要对哪些Joinpoint进行拦截的定义。
      Advice(通知/增强):所谓通知是指拦截到Joinpoint之后所要做的事情就是通知。通知分为前置通知(在方法之前执行)、后置通知(在方法之后执行)、异常通知(方法出现异常)、最终通知(在后置之后执行)、环绕通知(在方法之前和之后执行)。想要计算方法的执行时间:在方法之前执行,在方法之后执行,两个相减即可得到执行的时间。运用的就是环绕通知。
       Aspect(切面):是切入点和通知(引介)的结合。
      Introduction(引介):引介是一种特殊的通知在不修改类代码的前提下, Introduction可以在运行期为类动态地添加一些方法或Field。
      Target(目标对象):代理的目标对象(要增强的类)
      Weaving(织入):是把增强应用到目标的过程。把advice 应用到 target的过程
      Proxy(代理):一个类被AOP织入增强后,就产生一个结果代理类
      总结:
      切面和织入的区别:切面是将增强应用到具体方法上的过程,织入是将增强应用到具体的类上的过程。
     

    AOP原理图解

    对于有接口的情况:

    对于没有接口的情况:

    Spring中的AOP

      AOP并不是Spring框架特有的,Spring只是支持AOP编程的框架之一,SpringAOP是一种基于方法拦截的AOP,在Spring中有四种方式去实现AOP的拦截功能。

      1)使用 ProxyFactoryBean 和对应的接口实现AOP

      2)使用 XML 配置 AOP

      3)使用@AspectJ 注解驱动切面

      4)使用AspectJ 注入切面

      在Spring AOP的拦截方式中,真正常用的是用 @AspectJ 注解的方式实现的切面。

    使用表达式配置切入点

      1 切入点:实际增强的方法
      2 常用的表达式
        execution(<访问修饰符>?<返回类型><方法名>(<参数>)<异常>)
        (1)execution(* cn.itcast.aop.Book.add(..))           这个类中的这个方法
        (2)execution(* cn.itcast.aop.Book.*(..))               这个类中的所有方法
        (3)execution(* *.*(..))                                            所有类中的所有方法
      (4) 匹配所有save开头的方法 execution(* save*(..))
     
      3.指定切面的优先级
    • 在同一个连接点上应用不止一个切面时, 除非明确指定, 否则它们的优先级是不确定的.
    • 切面的优先级可以通过实现 Ordered 接口或利用 @Order 注解指定.
    • 实现 Ordered 接口, getOrder() 方法的返回值越小, 优先级越高.
    • 若使用 @Order 注解, 序号出现在注解中
      4.重用切入点定义
    • 在编写 AspectJ 切面时, 可以直接在通知注解中书写切入点表达式. 但同一个切点表达式可能会在多个通知中重复出现.
    • 在 AspectJ 切面中, 可以通过 @Pointcut 注解将一个切入点声明成简单的方法. 切入点的方法体通常是空的, 因为将切入点定义与应用程序逻辑混在一起是不合理的.
    • 切入点方法的访问控制符同时也控制着这个切入点的可见性. 如果切入点要在多个切面中共用, 最好将它们集中在一个公共的类中. 在这种情况下, 它们必须被声明为 public. 在引入这个切入点时, 必须将类名也包括在内. 如果类没有与这个切面放在同一个包中, 还必须包含包名.
    • 其他通知可以通过方法名称引入该切入点.

    使用@AspectJ 注解开发Spring AOP

    package com.cnblogs.demrystv.aop.aspect;
    
    import org.aspectj.lang.ProceedingJoinPoint;
    import org.aspectj.lang.annotation.After;
    import org.aspectj.lang.annotation.AfterReturning;
    import org.aspectj.lang.annotation.AfterThrowing;
    import org.aspectj.lang.annotation.Around;
    import org.aspectj.lang.annotation.Aspect;
    import org.aspectj.lang.annotation.Before;
    import org.aspectj.lang.annotation.DeclareParents;
    import org.aspectj.lang.annotation.Pointcut;
    
    import com.cnblogs.demrystv.aop.verifier.RoleVerifier;
    import com.cnblogs.demrystv.aop.verifier.impl.RoleVerifierImpl;
    import com.cnblogs.demrystv.game.pojo.Role;
    
    @Aspect
    public class RoleAspect {
        
        @DeclareParents(value= "com.cnblogs.demrystv.aop.service.impl.RoleServiceImpl+", defaultImpl=RoleVerifierImpl.class)
        public RoleVerifier roleVerifier;
    
        @Pointcut("execution(* com.cnblogs.demrystv.aop.service.impl.RoleServiceImpl.printRole(..))")
        public void print() {
        }
    
        @Before("print()")
        // @Before("execution(*
        // com.cnblogs.demrystv.aop.service.impl.RoleServiceImpl.printRole(..))")
        public void before() {
            System.out.println("before ....");
        }
    
        @After("print()")
        // @After("execution(*
        // com.cnblogs.demrystv.aop.service.impl.RoleServiceImpl.printRole(..))")
        public void after() {
            System.out.println("after ....");
        }
    
        @AfterReturning("print()")
        // @AfterReturning("execution(*
        // com.cnblogs.demrystv.aop.service.impl.RoleServiceImpl.printRole(..))")
        public void afterReturning() {
            System.out.println("afterReturning ....");
        }
    
        @AfterThrowing("print()")
        // @AfterThrowing("execution(*
        // com.cnblogs.demrystv.aop.service.impl.RoleServiceImpl.printRole(..))")
        public void afterThrowing() {
            System.out.println("afterThrowing ....");
        }
    
        @Around("print()")
        public void around(ProceedingJoinPoint jp) {
            System.out.println("around before ....");
            try {
                jp.proceed();
            } catch (Throwable e) {
                e.printStackTrace();
            }
            System.out.println("around after ....");
        }
    
        @Before("execution(* com.cnblogs.demrystv.aop.service.impl.RoleServiceImpl.printRole(..)) " + "&& args(role, sort)")
        public void before(Role role, int sort) {
            System.out.println("before ....");
        }
    }
     
  • 相关阅读:
    PCB 奥宝LDI 输出正负片转换关系
    PCB拼板之多款矩形排样算法实现--学习
    PCB 3D PCB 后续改进与扩展功能一些想法
    PCB 周期日历
    PCB LDI文件 自动化输出(改造)实现思路
    PCB Windows远程桌面一键登录
    PCB MS SQL 排序应用---SQL相邻数据区间值求解
    PCB MS SQL 排序应用---相邻数据且相同合并处理
    SpringMVC快速入门
    linux安装jdk
  • 原文地址:https://www.cnblogs.com/Demrystv/p/9271388.html
Copyright © 2020-2023  润新知