详解使用Spring Boot的AOP处理自定义注解
上一篇文章Java注解介绍讲解了下Java注解的基本使用方式,并且通过自定义注解实现了一个简单的测试工具;本篇文章将介绍如何使用SpringBoot的AOP来简化处理自定义注解,并将通过实现一个简单的方法执行时间统计工具为样例来讲解这些内容。
AOP概念
面向侧面的程序设计(aspect-orientedprogramming,AOP,又译作面向方面的程序设计、观点导向编程、剖面导向程序设计)是计算机科学中的一个术语,指一种程序设计范型。该范型以一种称为侧面(aspect,又译作方面)的语言构造为基础,侧面是一种新的模块化机制,用来描述分散在对象、类或函数中的横切关注点(crosscuttingconcern)。
侧面的概念源于对面向对象的程序设计的改进,但并不只限于此,它还可以用来改进传统的函数。与侧面相关的编程概念还包括元对象协议、主题(subject)、混入(mixin)和委托。
注释:以上定义源自中文维基百科(如果访问不了,可以通过修改系统的hosts文件访问,198.35.26.96zh.wikipedia.org#中文维基百科,只能帮到这了,如果还是上不了,那就麻烦上网搜索下怎么修改系统的hosts文件,不同系统下hosts文件位置不一样,如果是Linux或者Mac系统,我就直接告诉你吧,一般文件路径是/etc/hosts),AOP这个词的翻译有点和国内主流叫法不一致,国内主流都把AOP译做「面向切面编程」,大家不要拘泥于叫法,知道指的是同一个东西即可。
估计,你看了这个定义也是懵的,如果想深入了解可以去知乎看看大佬们是如何掰扯的什么是面向切面编程AOP?。我这边还是就直接上例子了吧。
SpringBoot的AOP环境准备
在pom.xml中引入相应的依赖模块
org.springframework.boot spring-boot-starter-parent 1.5.1.RELEASE org.springframework.boot spring-boot-starter-aop org.springframework.boot spring-boot-starter-web
先实现一个简单的Web请求处理
一个简单的处理Web请求的Controller。
packagecom.craneyuan.controller; importcom.craneyuan.service.IHelloWorldService; importorg.springframework.beans.factory.annotation.Autowired; importorg.springframework.web.bind.annotation.RequestMapping; importorg.springframework.web.bind.annotation.RequestMethod; importorg.springframework.web.bind.annotation.RestController; @RestController publicclassHelloWorldController{ @Autowired privateIHelloWorldServicehelloWorldService; @RequestMapping(value="/hello",method=RequestMethod.GET) publicStringhello(Stringname){ returnhelloWorldService.getHelloMessage(name); } }
一个简单的HelloWorld服务实现类,接口的定义我就不展示代码了。
packagecom.craneyuan.service.impl; importcom.craneyuan.annotation.AnalysisActuator; importcom.craneyuan.service.IHelloWorldService; importorg.springframework.beans.factory.annotation.Value; importorg.springframework.stereotype.Service; importjava.util.Optional; @Service publicclassHelloWorldServiceImplimplementsIHelloWorldService{ publicStringgetHelloMessage(Stringname){ return"Hello"+Optional.ofNullable(name).orElse("World!"); } }
这样一个简单的Web服务就弄好了,你可以启动项目用curl命令调用试下,例如:curl-XGET-i"http://127.0.0.1:8080/hello?name=Java",如果一切顺利的话,你将会得到类似下面这样的响应:
HTTP/1.1200 Content-Type:text/plain;charset=UTF-8 Content-Length:11 Date:Thu,11Jan201809:45:38GMT HelloJava
使用自定义注解来统计方法的执行时间
先定义一个用来统计方法执行时间的注解。
packagecom.craneyuan.annotation; importjava.lang.annotation.ElementType; importjava.lang.annotation.Retention; importjava.lang.annotation.RetentionPolicy; importjava.lang.annotation.Target; @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public@interfaceAnalysisActuator{ Stringnote()default""; }
然后定义一个切面,来处理刚刚定义的注解。
packagecom.craneyuan.aspect; importcom.craneyuan.annotation.AnalysisActuator; importorg.aspectj.lang.JoinPoint; importorg.aspectj.lang.annotation.After; importorg.aspectj.lang.annotation.Aspect; importorg.aspectj.lang.annotation.Before; importorg.aspectj.lang.annotation.Pointcut; importorg.slf4j.Logger; importorg.slf4j.LoggerFactory; importorg.springframework.core.annotation.Order; importorg.springframework.stereotype.Component; @Aspect @Component publicclassAnalysisActuatorAspect{ finalstaticLoggerlog=LoggerFactory.getLogger(AnalysisActuatorAspect.class); ThreadLocalbeginTime=newThreadLocal<>(); @Pointcut("@annotation(analysisActuator)") publicvoidserviceStatistics(AnalysisActuatoranalysisActuator){ } @Before("serviceStatistics(analysisActuator)") publicvoiddoBefore(JoinPointjoinPoint,AnalysisActuatoranalysisActuator){ //记录请求到达时间 beginTime.set(System.currentTimeMillis()); log.info("cy666note:{}",analysisActuator.note()); } @After("serviceStatistics(analysisActuator)") publicvoiddoAfter(AnalysisActuatoranalysisActuator){ log.info("cy666statistictime:{},note:{}",System.currentTimeMillis()-beginTime.get(),analysisActuator.note()); } }
最后,只要在需要统计执行时间的方法上加上@AnalysisActuator注解就行了。
packagecom.craneyuan.service.impl; importcom.craneyuan.annotation.AnalysisActuator; importcom.craneyuan.service.IHelloWorldService; importorg.springframework.beans.factory.annotation.Value; importorg.springframework.stereotype.Service; importjava.util.Optional; @Service publicclassHelloWorldServiceImplimplementsIHelloWorldService{ @AnalysisActuator(note="获取聊天信息方法") publicStringgetHelloMessage(Stringname){ return"Hello"+Optional.ofNullable(name).orElse("World!"); } }
启动项目,用curl命令随便调用一下,如果顺利的话就可以观察到切面打印的日志了。
... cy666statistictime:4,note:获取聊天信息方法
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。