SpringAOP的两种实现方式

1、SpringAOP,面向切面编程。在实际应用汇总很常见,一般用于日志、异常保存。也可以针对于相应的业务做处理

AOP核心概念

  1、横切关注点

  对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点

  2、切面(aspect)

  类是对物体特征的抽象,切面就是对横切关注点的抽象

  3、连接点(joinpoint)

  被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器

  4、切入点(pointcut)

  对连接点进行拦截的定义

  5、通知(advice)

  所谓通知指的就是指拦截到连接点之后要执行的代码,通知分为前置、后置、异常、最终、环绕通知五类

  6、目标对象

  代理的目标对象

  7、织入(weave)

  将切面应用到目标对象并导致代理对象创建的过程

  8、引入(introduction)

  在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段

2、切面的配置方式有两种:xml配置(建议)、注解

3、第一种方式:注解

  1)在相应的包里面建好自己类

  2)在相关的类上面加入相关注解,目的是认为这个类是切面

    

  3)进行相关切面的处理。这里存在两种方式,一种是直接类的方式,一种是用注解的方式

  这是第一种方式,是直接对应相关的类里面的方法做处理

package com.troy.ai.web.Interceptors;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;

@Aspect
@Component
public class AOPTEST {
    @Pointcut("execution(* com.troy.ai.web.schedule.ScheduleController.*(..))")
    public void test(){}
    
    @Before("test()") //也可以@Before("execution(* com.troy.ai.web.schedule.ScheduleController.*(..))")
    public void before(JoinPoint joinPoint){
        System.out.println(joinPoint.getSignature().getName()+"before");
    }
    
    @After("test()")
    public void after(JoinPoint joinPoint){
        System.out.println(joinPoint.getSignature().getName()+"after");
    }
}

  第二种:是通过注解的方式实现

  a、自己编写一个注解:

import java.lang.annotation.*;

@Target({ElementType.PARAMETER,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface SystemLog {
    String  description()  default "";
}

  b、再通过注解配置的方式来实现切面

@Aspect
@Component
public class LogInterceptor {
    private long  startTime;
    private JSONObject json = new JSONObject();
    @Pointcut("@annotation(com.troy.ai.web.Interceptors.SystemLog)")  
    public void  logAspect(){}
    @Before("logAspect()")
    public void before(JoinPoint JoinPoint){}

    @AfterThrowing("logAspect()")  
    public void  afterThrowing(){}

    @After("logAspect()")
    public void  after(JoinPoint JoinPoint){}

  这样只需要在要切面的类的方法上面加入这个注解就可以了

  

  这样这个方法,就可以通过自己的逻辑实现相关业务

  3)解释:JoinPoint 是对应切面所提供的接口,可以通过JoinPoint 来查找具体的信息

  

4、xml的配置方式,其实原理差不多

  1)还是写好对应的切面类(不需要做任何注解)

import org.aspectj.lang.JoinPoint;
public class AOPTEST {

    public void before(JoinPoint joinPoint){
        System.out.println(joinPoint.getSignature().getName()+"before");
    }
    
    public void after(JoinPoint joinPoint){
        System.out.println(joinPoint.getSignature().getName()+"after");
    }
}

  2)进行xml配置

<bean id="AOPTEST" class="com.troy.ai.web.Interceptors.AOPTEST"/>
    <aop:config>
        <aop:aspect id="aspect" ref="AOPTEST">
            <aop:pointcut expression="execution(* com.troy.ai.web.schedule.ScheduleController.*(..))" id="pointcut"/>
            <aop:before method="before" pointcut-ref="pointcut"/>
            <aop:after method="after" pointcut-ref="pointcut"/>
        </aop:aspect>
    </aop:config>

  3)这种方式也可以使用注解的方式来实现

5、区别

  1)共同点:在效果过实现上面都是一样的,并且两种方式在引用过程中都很容易理解

  2)不同点:

    a、注解的方式方便代码书写,并且很容易针对于新增的方法做处理,但是不好的地方在于,注解很容易分散,不易管理

    b、xml的方式再是集中管理,不管业务逻辑关系在正在怎么变化,所有东西都统一配置,便于管理。相对的在使用上面就没有那么舒服了,在新增方法上面或者个别上面,会存在多配置的情况

posted @ 2017-03-03 10:03  小不点丶  阅读(504)  评论(0编辑  收藏  举报