大家好,我是执梗。本专栏将从Spring
入门开始讲起,详细讲解各类配置的使用以及原因,到使用SpringBoot
进行开发实战,旨在记录学习生活的同时也希望能帮到大家,如果对您能有所帮助,还望能点赞关注该专栏,对于专栏内容有错还望您可以及时指点,非常感谢大家 🌹。
AOP
:面向切面编程,主要是一种思想,将某一类事情来进行集中的处理。 可以减少冗余代码,而且提高代码的解耦合,可重用性以及可维护性。
SpringAOP
则是AOP
思想的一种具体实现,它是一个框架,通过它我们可以利用AOP
思想来设计代码。
举一个常见的场景,我们在做后台系统时,除了登录和注册两个板块是不需要登录验证即可访问之外,其余的页面都应该必须先验证用户登录状态,如果未登录是不允许访问的。如果按照老版本的做法,对于每一个 Controller
我们都需要去验证用户登录状态,那么随着工程的壮大,这种判断代码会越来越多,我们维护的成本或会越来越高。
这种应用单一,且需要被多处调用的功能,就可以考虑 AOP 来统一进行处理达到优化目的。
除了登录验证这个功能外,AOP还可以解决以下这些场景:
由于都是英语直译过来,下面的名词按中文理解可能会有点抽象,不过大家知道是啥作用即可。
切面是横切关注点的实现。在 Spring AOP
中,一个切面通常是一个 Java
类,它包含一些通知(Advice)和一个切入点(Pointcut)。
切面可以理解为 AOP 实现某个功能的集合
连接点是程序执行过程中能够插入切面的点。在 Spring AOP
中,连接点通常是方法调用或方法执行的时刻。
连接点相当于需要被增强的某个 AOP 功能的所有⽅法。
切入点定义了哪些连接点会触发通知。它使用切入点表达式来匹配连接点。
切点相当于保存了众多连接点的⼀个集合(如果把切点看成⼀个表,而连接点就是表中⼀条⼀条的数据)
通知是切面在连接点处执行的代码。 在SpringAOP
中有五种类型的通知,通常我们可以在方法上使用注解来使方法称为对应的通知方法:
@Before
:通知方法会在目标方法返回后调用。@After
:通知方法会在目标方法抛出异常后调用。@AfterReturning
:通知方法会在目标方法返回后调用@AfterThrowing
:通知方法会在目标方法抛出异常后调用。@Around
:通知包裹了被通知的方法,在被通知的方法通知之前和调用之后执行自定义的行为 接下来我们来初步实现以下 AOP的 功能,目标是去拦截 UserController
里面的方法,并执行相对应的通知事件。
在 pom.xml
添加依赖:
org.springframework.boot spring-boot-starter-aop
切点指的是我们需要处理的某一类问题,比如用户权限验证就是一个问题,切点定义的是某一类问题
SpringAOP
切点的定义如下,在切点中需要定义拦截规则,代码实现如下:
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;@Aspect //表明当前类是一个切面
@Component
public class UserAspect {//定义切点(设置拦截规则)@Pointcut("execution(* com.example.springbootaop.controller.UserController.*(..))")public void pointcut() {}/** 定义poincut切点的前置通知* */@Before("pointcut()")public void doBefore() {System.out.println("前置通知被执行了");}/** return 之前返回* */@AfterReturning("pointcut()")public void doAfterReturning(){System.out.println("执行了 doAfterReturning 方法");}/** 后置通知* */@After("pointcut()")public void doAfter() {System.out.println("后置通知被执行了");}
}
注意这里的注解@Pointcut
里的参数是我们需要拦截的路径,是AspectJ
表达式的写法,这里不过多讲解。
配置Controller
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
@RequestMapping("/user/")
public class UserController {@RequestMapping("/sayhi")public String sayHi(){System.out.println("sayhi 方法被执行了");return "你好,世界";}
}
当我们访问后,回答控制台查看打印:
从打印顺序来看是符合预期的。
@AfterThrowing
会在抛出异常的时候执行,我们
@AfterThrowing("pointcut()")public void doAfterThrowing() {System.out.println("执行了 doAfterThrowing 方法");}
更改Controller
里的代码,使其产生一个除零异常我们来测试以下:
@RestController
@RequestMapping("/user/")
public class UserController {@RequestMapping("/sayhi")public String sayHi(){int a=10/0;return "你好,世界";}
}
这里我们发现,即使出现异常,后置方法仍然会执行。
SpringAOP
是构建在动态代理基础上,因此它仅支持局限于方法级别的拦截
下一篇:传输安全HTTPS