Spring 框架学习(八)——AOP 的认识与使用

Spring 框架学习(八)——AOP的认识与使用


一、Spring AOP 常用概念及铺垫


先说几个常用的概念以及铺垫的知识


横切点:就是我们要给方法前加一个打印日志的功能,或者先校验等等,这些功能

切面(Aspect):将横切关注点进行模块化的一个类

通知(Adviser):就是切面里面具体的方法,用来切入到具体位置的方法里面

切入点(PointCut):一个业务类中的具体方法,就是把通知切入到这个方法里面

连接点(JoinPoint):通过连接点我们可以知道切入点方法对象的很多信息。

在这里插入图片描述


二、通知的介绍


通知类型就是想要加的代码(校验、日志等) 是在对象方法的前面还是后面执行的类型,这就是通知类型。


(1)before 前置通知

在方法执行前执行

如果方法出现了异常,不会影响前置通知的执行

应用:通常应用在执行方法前各种校验


(2)after 后置通知

在方法执行完毕之后执行

无论方法是否出现异常终止都会执行

应用:清理现场,关闭、释放资源


(3)around 环绕通知

方法执行前后分别执行

如果方法中出现异常终止,那么末尾的通知就不执行了

应用:各个方面,结合了前置和后置的优点,还可以拿到对象方法的各种参数及信息


(4)afterReturning 返回后通知

方法正常返回后执行

如果方法抛出异常,那么无法执行

应用:常规结果数据处理


(5)afterThrowing 抛出异常后通知

方法抛出异常后执行

如果方法没有抛出异常,无法执行

应用:抛出异常后对信息进行包装


三、切点表达式说明


(1)作用


用来匹配具体切入点(方法)的具体位置


(2)格式


execution(切点表达式)

execution([修饰符] 返回类型 包名.类名.方法名(参数类型)

方法访问修饰符可以进行省略,但是后面的东西必须得有


(3)通配符介绍

  • *(一个星号): 可以匹配任意 返回值、包名、类名、方法名、参数类型

  • …(两个点):表示任意多个层级、任意多个参数

具体使用

我们想要把切点定义到 com.bit.service 包下的 UserService 类中的 void add(int a) 方法

有很多种写法


1.每一部分都写的具体

  execution("void com.bit.service.UserService.add(int)")

2.完全使用通配符

  execution("* *..service.*(..)")
  //* 任意返回类型
  //* 任意一个包
  // .. 包后面跟多层
  // * 类名
  // * 方法名
  //(..)任意多个任意参数
  execution("* *..*.*(..)")
  //* 任意返回类型
  //* 任意一个包
  // .. 包后面跟多层
  // * 类名
  // * 方法名
  //(..)任意多个任意参数
  execution("* *..*(..)")
  //* 任意返回类型
  //* 任意一个包
  // .. 包后面跟多层包+类名(接口名)
  // * 方法名
  //(..)任意多个任意参数

四、连接点及环绕方法参数使用


前置通知,写一个方法,在切入点之前执行通知即可

后置通知,写一个方法,在切入点之前执行通知即可

那么环绕通知的代码如何进行环绕呢?

这里就要用到 连接点(JoinPoint) 的一些使用了


(1) JoinPoint 作为环绕通知 的方法参数


JoinPoint对象封装了SpringAop中切面方法的信息,在切面方法中添加JoinPoint参数,就可以获取到封装了该方法信息的JoinPoint对象.

JoinPoint 只有获取切入点对象方法信息的一些功能,不能帮助我们进行环绕写代码

在这里插入图片描述

能够获取对象方法参数、方法签名等信息


(2) ProceedingJoinPoint 作为环绕通知 的方法参数


ProceedingJoinPoint对象是JoinPoint的子接口,该对象只用在@Around的切面方法中,
添加了

Object proceed() throws Throwable //执行目标方法

Object proceed(Object[] var1) throws Throwable //传入的新的参数去执行目标方法
两个方法.

在这里插入图片描述

调用proceed方法,相当于执行切入点方法


(3) 环绕通知的写法

    public void around(ProceedingJoinPoint joinPoint) throws Throwable {
    
    
        System.out.println("Around=====方法环绕前====");
        joinPoint.proceed();  // 执行方法
        System.out.println("Around======方法环绕后=====");
    }

(4) 在执行方法的时候修改方法参数


1、使用 Object [] args 接收原方法传入的参数

 Object[] args = joinPoint.getArgs();

2、对数组中的参数进行修改

  for (int i = 0; i <args.length ; i++) {
    
    
        args[i] = (Integer)args[i]+10;
  }

3、执行proceed带参数的方法,将修改过的object[] args进行传入

joinPoint.proceed(args);  // 执行方法

五、Spring-AOP 准备工作


(1)加入aspect织入包


在编写SpringAOP面向切面编程时,需要导入一个aspectjweaver.jar的包,它的主要作用是负责解析切入点表达式。

    <!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.9.1</version>
        </dependency>

(2)加入aop约束以及开启aop注解支持


可以在xml使用<aop:config 回车自动生成,下面是aop约束已经加上了的

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--开启aop注解的支持 -->
    
    <aop:aspectj-autoproxy />
    
    <!--在aop:config 中配置切面、切点、以及各种通知 或者使用注解配置即可-->

    <context:annotation-config/>

    <context:component-scan base-package="com.*"/>
</beans>

六、Spring中如何使用AOP


(1)xml配置使用AOP


(1)先写一个业务的接口

package com.service;

public interface UserService {
    
    
    void add();
};


(2)给这个接口创建创建一个实现类,类中的方法作为切入点。

package com.service;
import org.springframework.stereotype.Service;

@Service
public class UserServiceImpl implements UserService {
    
    
    public void add() {
    
    
        System.out.println("执行了add方法");
    }
}

(3)自定义一个类作为切面,在里面实现一些方法作为具体的通知。

package com.config;
import org.aspectj.lang.ProceedingJoinPoint;

public class DiyAspect {
    
    
    // 该类作为切面,所有横切关注点的集合的一个模块

     public void before(){
    
    
         System.out.println("Before=====方法执行前====");
     }

    public void after(){
    
    
        System.out.println("After=====方法执行后====");
    }

    public void around(ProceedingJoinPoint pj) throws Throwable {
    
    
        System.out.println("Around=====方法环绕前====");
        pj.proceed();  // 执行方法
        System.out.println("Around======方法环绕后=====");
    }
    
}

(4)通过xml配置定义切点、切面、通知,以及使得切面里的通知绑定到切点上。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">

    <bean id="diyAspect" class="com.config.DiyAspect"/>
    
    <!--开启aop注解的支持 -->
    <aop:aspectj-autoproxy />

    <!--在aop:config 中配置切面、切点、以及各种通知 或者使用注解配置即可-->
    <aop:config >
        <aop:pointcut id="pointcut" expression="execution(* *..service.*.*(..))"/>
        <aop:aspect id="diy" ref="diyAspect">
            <aop:before method="before" pointcut-ref="pointcut"/>
            <aop:around method="around" pointcut-ref="pointcut"/>
            <aop:after method="after" pointcut-ref="pointcut"/>
        </aop:aspect>
    </aop:config>

<!--    开启组件的注解支持-->
    <context:annotation-config/>
<!--    开启包路径下的组件扫描-->
    <context:component-scan base-package="com.*"/>
</beans>

(5)进行测试,执行add方法,查看通知增强是否绑定切入点成功

package com.controller;

import com.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.stereotype.Controller;

@Controller
public class UserController {
    
    
    public static void main(String[] args) {
    
    
        ApplicationContext context =new ClassPathXmlApplicationContext("spring-config.xml");
        UserService userService = context.getBean("userServiceImpl", UserService.class);
        userService.add();
    }
}

(6)我们可以确定,通过xml注解的方式,前置通知、后置通知、环绕通知的执行顺序

在这里插入图片描述

环绕前和环绕后被 before after包裹着


(2)注解开发使用AOP


(1)使用注解 @Aspect 将自定义类作为切面,@PointCut 定义切点的位置,@Before / @After 定义通知作用到切点上

package com.config;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

@Aspect
@Component
public class DiyAspect {
    
    
    // 该类作为切面,所有横切关注点的集合的一个模块

    // 可以使用@PointCut 定义一个切点,在之后只需要调用这个方法即可,不需要重复的写切点表达式
    @Pointcut("execution( * com.service.UserServiceImpl.add(..))")
    public void pointCut(){
    
    

    }

    @Before("pointCut()")
     public void before(){
    
    
         System.out.println("Before=====方法执行前====");
     }

     @After("pointCut()")
    public void after(){
    
    
        System.out.println("After=====方法执行后====");
    }

    @Around("pointCut()")
    public void around(ProceedingJoinPoint joinPoint) throws Throwable {
    
    
        System.out.println("Around=====方法环绕前====");
        joinPoint.proceed();  // 执行方法
        System.out.println("Around======方法环绕后=====");
    }

}

(2)xml文件中可以完全不同写aop:config 的配置内容,但是必须加上aop约束和aop注解支持

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--开启aop注解的支持 -->
    <aop:aspectj-autoproxy />

<!--    开启组件的注解支持-->
    <context:annotation-config/>
<!--    开启包路径下的组件扫描-->
    <context:component-scan base-package="com.*"/>

</beans>

(3)进行测试,执行add方法,查看通知增强是否绑定切入点成功

package com.controller;

import com.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.stereotype.Controller;

@Controller
public class UserController {
    
    
    public static void main(String[] args) {
    
    
        ApplicationContext context =new ClassPathXmlApplicationContext("spring-config.xml");
        UserService userService = context.getBean("userServiceImpl", UserService.class);
        userService.add();
    }
}

(4) 我们可以确定,通过注解开发的方式,前置通知、后置通知、环绕通知的执行顺序和之前xml配置就不一样了

在这里插入图片描述

环绕通知在 before 和 after的外面包裹

猜你喜欢

转载自blog.csdn.net/rain67/article/details/125353742