AOP运行过程解析

来源:互联网 发布:js根据name获取标签 编辑:程序博客网 时间:2024/06/06 03:01

上篇写了动态代理,这里我们来看看AOP吧……

AOP面向切面编程的核心思想就是动态代理

想要应用AOP,主要有如下步骤:

1、编写一个切面类,在类中写一些方法,在些方法上配置连接点(Before、After、Around、AfterRuturning、AfterThrowing),在连接点后写明切点("excecution( * com.yc.demo..*.*(..))")第一个*表示任何类型的修饰符(public)和返回类型(int),第二个*表示类名,第三个*表示任何方法,..表示当前包或子包中,(..)表示任意个参数并且是任意类型,最后记住要在切面类上配置@Aspect,@Component,@Order(1)(指定切面的先后顺序,数字越小:该切面的@before方法在所有切面中越先执行,数字越大:该切面的@After方法在所有切面中越先执行)

2、在方法中写上想要执行的操作,此方法就会在匹配的切点上根据连接点的时机来执行。

3、其实还要配置xml文件才能运行呢:

在spring.xml文件中配置如下信息:

<context:component-scan base-package="com.yc.spring"/>       -------------扫包 ,可将实体类作为bean对象存于spring容器,按类型匹配

<aop:aspectj-autoproxy/>     -------------自动扫描aspect,然后与匹配的Bean创建代理

4、小伙伴们此时,会发现还要导入spring-aspects包来生成代理对象,当然咯,要引入spring.xml还要spring-context-support包,在spring.xml中还要收入aop和context的命名空间咯。


再加上几个名词解释加深理解吧:

    切面(Aspect):  横切关注点(跨越应用程序多个模块的功能)被模块化的特殊对象

    通知(Advice):  切面必须要完成的工作

    目标(Target): 被通知的对象

    代理(Proxy): 向目标对象应用通知之后创建的对象

    连接点(Joinpoint):程序执行的某个特定位置:如类某个方法调用前、调用后、方法抛出异常后等。连接点由两个信息确定:方法表示的程序执行点;相对点表示的方位。例如 ArithmethicCalculator#add() 方法执行前的连接点,执行点为 ArithmethicCalculator#add(); 方位为该方法执行前的位置

    切点(pointcut):每个类都拥有多个连接点:例如 ArithmethicCalculator 的所有方法实际上都是连接点,即连接点是程序类中客观存在的事务。AOP 通过切点定位到特定的连接点。类比:连接点相当于数据库中的记录,切点相当于查询条件。切点和连接点不是一对一的关系,一个切点匹配多个连接点,切点通过 org.springframework.aop.Pointcut 接口进行描述,它使用类和方法作为连接点的查询条件。


@Before: 前置通知, 在方法执行之前执行
            前置通知:在方法执行之前执行的通知
            前置通知使用 @Before 注解, 并将切入点表达式的值作为注解值.

        @After: 后置通知, 在方法执行之后执行
            后置通知是在连接点完成之后执行的, 即连接点返回结果或者抛出异常的时候, 下面的后置通知记录了方法的终止.
            一个切面可以包括一个或者多个通知.
            无论连接点是正常返回还是抛出异常, 后置通知都会执行. 如果只想在连接点返回的时候记录日志, 应使用返回通知代替后置通知.

        @AfterRunning: 返回通知, 在方法返回结果之后执行
            在返回通知中, 只要将 returning 属性添加到 @AfterReturning 注解中, 就可以访问连接点的返回值. 该属性的值即为用来传入返回值的参数名称.
            必须在通知方法的签名中添加一个同名参数. 在运行时, Spring AOP 会通过这个参数传递返回值.
            原始的切点表达式需要出现在 pointcut 属性中

        @AfterThrowing: 异常通知, 在方法抛出异常之后
            只在连接点抛出异常时才执行异常通知
            将 throwing 属性添加到 @AfterThrowing 注解中, 也可以访问连接点抛出的异常. Throwable 是所有错误和异常类的超类. 所以在异常通知方法可以捕获到任何错误和异常.
            如果只对某种特殊的异常类型感兴趣, 可以将参数声明为其他异常的参数类型. 然后通知就只在抛出这个类型及其子类的异常时才被执行.

        @Around: 环绕通知, 围绕着方法执行
            环绕通知是所有通知类型中功能最为强大的, 能够全面地控制连接点. 甚至可以控制是否执行连接点.
            对于环绕通知来说, 连接点的参数类型必须是 ProceedingJoinPoint . 它是 JoinPoint 的子接口, 允许控制何时执行, 是否执行连接点.
            在环绕通知中需要明确调用 ProceedingJoinPoint 的 proceed() 方法来执行被代理的方法. 如果忘记这样做就会导致通知被执行了, 但目标方法没有被执行.
            注意: 环绕通知的方法需要返回目标方法执行之后的结果, 即调用 joinPoint.proceed(); 的返回值, 否则会出现空指针异常

0 0
原创粉丝点击