亲宝软件园·资讯

展开

浅析Spring基于注解的AOP

学习使我快乐T 人气:0

一、准备工作

①创建一个Maven工程

②添加依赖

在IOC所需依赖基础上再加入下面依赖即可:

<!-- spring-aspects会帮我们传递过来aspectjweaver -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aspects</artifactId>
<version>5.3.1</version>
</dependency>

③把上节的接口和实现类复制过来,因为我们要在这个环境里面测试

二、基于注解的AOP之前置通知

如果我们要实现AOP的话,它也是要在我们的IOC的基础上实现的,所以说我们必须要把切面还有目标对象交给IOC容器来管理

AOP 抽横切关注点(非核心业务代码)

①创建切面类并配置

/**
 * 在切面中,需要通过指定的注解将方法标识为通知方法
 * @Before:前置通知,在目标对象方法执行之前执行
 */
@Component
@Aspect //将当前组件标识为切面
public class LoggerAspect {
    @Before("execution(public int com.tian.spring.aop.annotation.CalculatorImpl.add(int,int))")
    public void beforeAdviceMethod() {
        System.out.println("LoggerAspect,前置通知");
    }
}

②创建Spring配置文件,让ioc对我们的目标对象进行管理

    <!--
        AOP的注意事项:
        切面类和目标类都需要交给IOC容器管理
        切面类必须通过@Aspect注解标识为一个切面
        在Spring的配置文件中设置<aop:aspectj-autoproxy />开启基于注解的AOP
    -->
    <context:component-scan base-package="com.tian.spring.aop.annotation"></context:component-scan>
    <!--开启基于注解的AOP-->
    <aop:aspectj-autoproxy/>

测试类:

public class AOPTest {
    @Test
    public void testAOPByAnnotation() {
        ApplicationContext ioc = new ClassPathXmlApplicationContext("aop-annotation.xml");
        Calculator calculator = ioc.getBean(Calculator.class);
        calculator.add(1,2);
    }
}

三、基于注解的AOP之切入点表达式的语法和重用以及获取连接点的信息

1.在切面中,需要通过指定的注解将方法标识为通知方法

@Before:前置通知,在目标对象方法执行之前执行

2.切入点表达式:设置在标识通知的value属性中

execution(public int com.tian.spring.aop.annotation.CalculatorImpl.add(int,int))

execution(* com.tian.spring.aop.annotation.CalculatorImpl.*(..))

第一个*表示任意的访问修饰符和返回值类型

第二个*表示类中任意的方法

..表示任意的参数列表

类的地方也可以使用*,表示包下所有的类

①切入点表达式的语法

上述已经将我们的前置通知,通过切入点表达式作用到了我们的连接点上,下面我们来说是细节问题,因为我们设置得还不是很完美,就比如上面实现了后只能作用与我们的add方法,因为我们的切入点表达式是写死了的,下面我就来进行更完美的代码实现

将切面类中的切入点表达式修改为如下

/**
 * 
 * 切入点表达式:设置在标识通知的value属性中
 * execution(public int com.tian.spring.aop.annotation.CalculatorImpl.add(int,int))
 * execution(* com.tian.spring.aop.annotation.CalculatorImpl.*(..))
 * 第一个*表示任意的访问修饰符和返回值类型
 * 第二个*表示类中任意的方法
 * ..表示任意的参数列表
 * 类的地方也可以使用*,表示包下所有的类
 */
@Component
@Aspect //将当前组件标识为切面
public class LoggerAspect {
//    @Before("execution(public int com.tian.spring.aop.annotation.CalculatorImpl.add(int,int))")
    @Before("execution(* com.tian.spring.aop.annotation.*.*(..))")
    public void beforeAdviceMethod() {
        System.out.println("LoggerAspect,前置通知");
    }
}

测试类:

public class AOPTest {
    @Test
    public void testAOPByAnnotation() {
        ApplicationContext ioc = new ClassPathXmlApplicationContext("aop-annotation.xml");
        Calculator calculator = ioc.getBean(Calculator.class);
        calculator.sub(1,2);
    }
}

②获取连接点的信息

获取连接点的信息

在通知方法的参数位置,设置JoinPoint类型的参数,就可以获取连接点所对应方法的信息 获取连接点所对应方法的签名信息

Signature signature = joinPoint.getSignature();

获取连接点所对应方法的参数

Object[] args = joinPoint.getArgs();

System.out.println("LoggerAspect,方法:" + signature.getName() + ",参数" + Arrays.toString(args));

我们在之前的动态代理里面,我们在前置通知的位置,也就是在我们目标对象的方法执行之前,然后我们在方法体中输出的是我们要调用的方法的方法名,还有就是我们当前的参数列表,但是我们用了前置通知之后,我们不知道如何获取了,还没有我们动态代理实现的功能多。而且连接点的信息我们都获取不到,也就是我当前要加入通知的方法,它的一些信息我们都获取不到。其实我们也是可以获取到的,下面我就来实现

//    @Before("execution(public int com.tian.spring.aop.annotation.CalculatorImpl.add(int,int))")
//    @Before("execution(* com.tian.spring.aop.annotation.*.*(..))")
    @Pointcut("pointCut()")
    public void beforeAdviceMethod(JoinPoint joinPoint) {
        //获取连接点所对应方法的签名信息
        Signature signature = joinPoint.getSignature();
        //获取连接点所对应方法的参数
        Object[] args = joinPoint.getArgs();
        System.out.println("LoggerAspect,方法:" + signature.getName() + ",参数" + Arrays.toString(args));
    }

③重用写入点表达式

重用切入点表达式

//@Pointcut声明一个公共的切入点表达式

@Pointcut("execution(* com.tian.spring.aop.annotation.CalculatorImpl.*(..))") public void pointCut() {}

使用方式:@Pointcut("pointCut()")

声明

    @Pointcut("execution(* com.tian.spring.aop.annotation.CalculatorImpl.*(..))")
    public void pointCut() {
    }

引用

    @After("pointCut()")
    public void afterAdviceMethod() {
    }

加载全部内容

相关教程
猜你喜欢
用户评论