Spring Boot系列——AOP配自定义注解的最佳实践

AOP(Aspect Oriented Programming),即面向切面编程,是Spring框架的大杀器之一。

首先,我声明下,我不是来系统介绍什么是AOP,更不是照本宣科讲解什么是连接点、切面、通知和切入点这些让人头皮发麻的概念。

今天就来说说AOP的一些应用场景以及如何通过和其他特性的结合提升自己的灵活性。

AOP应用举例

AOP的一大好处就是解耦。通过切面,我们可以将那些反复出现的代码抽取出来,放在一个地方统一处理。

同时,抽出来的代码很多是与业务无关的,这样可以方便开发者更加专注自己的业务逻辑的开发。

一个AOP的典型应用场景就是日志打印。

下面是一个极端情况的Controller

@RestController @RequestMapping("http://www.likecs.com/") public class HelloController { private static final Logger LOG = LoggerFactory.getLogger(HelloController.class); @GetMapping(value = "/index") public String index(HttpServletRequest request) { LOG.info("============打印日志开始============"); LOG.info("URL: " + request.getRequestURL().toString()); LOG.info("============打印日志结束============"); return "hello jackie"; } @GetMapping(value = "/test1") public String test1(HttpServletRequest request, String var1) { LOG.info("============打印日志开始============"); LOG.info("URL: " + request.getRequestURL().toString()); LOG.info("============打印日志结束============"); return "test1"; } @DemoAnnotation @GetMapping(value = "/test2") public String test2(HttpServletRequest request, String var1, String var2) { LOG.info("============打印日志开始============"); LOG.info("URL: " + request.getRequestURL().toString()); LOG.info("============打印日志结束============"); // int i = 1/0; if (1<2) throw new IllegalArgumentException("exception"); return "test2"; } }

HelloController中提供了三个Http接口,由于业务需要,所以每次进入某个方法的时候都需要打印请求的相关信息。

当然,如果只是上面的例子,我们完全可以通过其他手段让代码看着并不这么糟糕。我们可以抽象一个打印方法,将相同的代码封装在这个方法中,之后在各个方法中每次调用即可。

但是,这种处理方法似乎抽象的还不够,因为我们在每个Http接口中还是要调用这个抽象的函数。而且,比较要命的是,这打印日志的代码与其他业务代码显得有些格格不入。

所以,这时候,我们想到了AOP。

如何使用AOP

在Spring Boot项目中,只需要如下几步,就可以轻松上手AOP。

添加maven依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency> 编写切面类 @Aspect @Component public class DemoAspect { private static final Logger LOG = LoggerFactory.getLogger(DemoAspect.class); @Pointcut("execution(public * com.jackie.springbootdemo.controller.HelloController.test*(..))") public void addAdvice(){} @Before("addAdvice()") public void before(JoinPoint joinPoint){ Object[] args = joinPoint.getArgs(); HttpServletRequest requests = (HttpServletRequest) args[0]; LOG.info("============打印日志开始============"); LOG.info("URL: " + requests.getRequestURL().toString()); LOG.info("============打印日志结束============"); // LOG.info("before...."); } } 结果验证对比

启动SpringBootDemoApplication,访问url::8080/test2?var1=1&var2=2

未使用切面功能打印日志

Spring Boot系列——AOP配自定义注解的最佳实践

使用切面功能打印日志

Spring Boot系列——AOP配自定义注解的最佳实践

从上面的结果展示发现,最终的效果是一样的,但是使用切面更加简洁,而且可复用。

如上访问的是test2接口,如果访问test1接口也可以走切面类实现打印日志的需求,但是如果走index请求就不会打印日志了。

这是为什么呢?

AOP的局限

在切面类DemoAspect中,我们看到了切入点的设置

@Pointcut("execution(public * com.jackie.springbootdemo.controller.HelloController.test*(..))") public void addAdvice(){}

其中Pointcut后面的表达式是用于控制切面的有效影响范围。

**表达式中,第一个表示返回任意类型,第二个表示任意方法名,后面的小括号表示任意参数值,这里是以test为前缀的,所以可以匹配上test1和test2方法。

注意,在第二个之前也可以再有个,即HelloController所在位置,表示任意类名,假如这里是有两个*.则表示包括包里面的子包。**

好了,明白了表达式的含义,我们自然就看到了AOP的局限性。

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/wpfdfj.html