学习Spring Boot:(二十二)使用 AOP
前言
AOP 1,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。基于AOP实现的功能不会破坏原来程序逻辑,因此它可以很好的对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
正文
Spring Boot 中使用
- 在
pom.xml
中加入 aop 依赖:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
- 当我们需要在非接口上面进行切面操作的时候,就需要
CGLIB
来实现 AOP,在系统配置文件中加入设置:
spring:
aop:
proxy-target-class: true
默认为 false
。
切点表达式
列出常用的几个表达式:
1. execution()
满足execution中描述的方法签名。 execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)throws-pattern?)
* modifier-pattern
:表示方法的修饰符;
* ret-type-pattern
:表示方法的返回值
* declaring-type-pattern
:表示方法所在的类的路径
* name-pattern
:表示方法名
* param-pattern
:表示方法的参数
* throws-pattern
:表示方法抛出的异常
* 其中后面跟着?
的是可选项。
this()
是用来限定方法所属的类,为接口则限定所有的实现类,为类的话,限定这单个类。@annotation
表示具有某个标注的方法。args
表示方法的参数属于一个特定的类,@args
表示参数有特定的标注注解。within
包或者类型满足within中描述的包或者类型的类的所有非私有方法,@within
类型拥有@target描述中给出的annotation,其中@target和@within的区别在于@within要求的annotation的级别为CLASS,而@target为RUNTIME
.target
业务实例对象(非代理实例)的类型满足target 中的描述的类型,@target 类型拥有@target描述中给出的annotationbean()
表示所有匹配的 bean,例如 ,bean(“*Service”),匹配所有Service
结尾的类。可以使用!bean()
表示不匹配。
注意事项:
* 在各个pattern中,可以使用”*”来表示匹配所有。
* 在param-pattern
中,可以指定具体的参数类型,多个参数间用,
隔开,各个也可以用 *
来表示匹配任意类型的参数,如(String)
表示匹配一个String
参数的方法;(*,String)
表示匹配有两个参数的方法,第一个参数可以是任意类型,而第二个参数是String类型。
* 可以用(..)
表示零个或多个任意的方法参数。
* 使用&&
符号表示与关系,使用||
表示或关系、使用!
表示非关系。在XML文件中使用and
、or
和not
这三个符号。
AspectJ提供了五种定义通知的标注:
@Before
:前置通知,在调用目标方法之前执行通知定义的任务@After
:后置通知,在目标方法执行结束后,无论执行结果如何都执行通知定义的任务@AfterReturning
:后置通知,在目标方法执行结束后,如果执行成功,则执行通知定义的任务@AfterThrowing
:异常通知,如果目标方法执行过程中抛出异常,则执行通知定义的任务@Around
:环绕通知,在目标方法执行前和执行后,都需要执行通知定义的任务
通过标注定义通知只需要两个步骤:
1. 将以上五种标注之一添加到切面的方法中
2. 在标注中设置切点的定义。
创建环绕通知
环绕通知相比其它四种通知有其特殊之处。环绕通知本质上是将前置通知、后置通知和异常通知整合成一个单独的通知。
用@Around
标注的方法,该方法必须有一个ProceedingJoinPoint
类型的参数,
在方法体中,需要通过这个参数,以joinPoint.proceed();
的形式调用目标方法。注意在环绕通知中必须进行该调用,否则目标方法本身的执行就会被跳过。
计算方法的执行时间:
@Around("logPointCut()") //切点
public Object around(ProceedingJoinPoint point) throws Throwable {
long beginTime = System.currentTimeMillis();
//执行方法
Object result = point.proceed();
//执行时长(毫秒)
long time = System.currentTimeMillis() - beginTime;
return result;
}
处理通知中参数
获取参数的方式则需要使用关键词是args
。
@Pointcut("bean(sysUserServiceImpl) && args(userEntity,..)")
public void userPointCut(SysUserEntity userEntity) {
}
@Before("userPointCut(userEntity)")
public void validateUser(SysUserEntity userEntity) {
// to handler args
}
这里有个非常严格的一点就是,args(userEntity,..)
,表示目标方法,可能有多个参数,但是包括 userEntity
,这里 userEntity
必须参数名相同,不同就编织了。
args()
中参数的名称必须跟切点方法的签名中public void validateUser(SysUserEntity userEntity)
的参数名称相同。如果使用切点函数定义,其中的参数名称也必须与通知方法签名中的参数完全相同
AfterReturning增强处理
@AfterReturning(pointcut = "logPointCut()", returning = "rtv")
public void logAfter(Object rtv) {
System.out.println(Objects.toString(rtv));
}
使用 @AfterReturning
注解时,指定了一个returning
属性,假设该属性值为rvt
,这表明允许在Advice方法(logAfter()方法)中定义名为rvt的形参,程序可通过rvt形参来访问目标方法的返回值。
注意:
虽然AfterReturning
增强处理可以访问到方法的返回值,但它不可以改变目标方法的返回值。
AOP切面的优先级
有时候,我们对一个方法会有多个切面的问题,这个时候还会涉及到切面的执行顺序的问题。
我们可以定义每个切面的优先级, Spring 中提供注解 @Order(i)
,当 i
的值越小,优先级越高。
参考文章
- Aspect Oriented Programming的缩写 ↩
学习Spring Boot:(二十二)使用 AOP的更多相关文章
- 学习Spring Boot:(二十五)使用 Redis 实现数据缓存
前言 由于 Ehcache 存在于单个 java 程序的进程中,无法满足多个程序分布式的情况,需要将多个服务器的缓存集中起来进行管理,需要一个缓存的寄存器,这里使用的是 Redis. 正文 当应用程序 ...
- spring boot / cloud (十二) 异常统一处理进阶
spring boot / cloud (十二) 异常统一处理进阶 前言 在spring boot / cloud (二) 规范响应格式以及统一异常处理这篇博客中已经提到了使用@ExceptionHa ...
- 学习Spring Boot:(二十六)使用 RabbitMQ 消息队列
前言 前面学习了 RabbitMQ 基础,现在主要记录下学习 Spring Boot 整合 RabbitMQ ,调用它的 API ,以及中间使用的相关功能的记录. 相关的可以去我的博客/RabbitM ...
- 学习 Spring Boot:(二十九)Spring Boot Junit 单元测试
前言 JUnit 是一个回归测试框架,被开发者用于实施对应用程序的单元测试,加快程序编制速度,同时提高编码的质量. JUnit 测试框架具有以下重要特性: 测试工具 测试套件 测试运行器 测试分类 了 ...
- 学习Spring Boot:(二十四)多数据源配置与使用
前言 随着业务量增大,可能有些业务不是放在同一个数据库中,所以系统有需求使用多个数据库完成业务需求,我们需要配置多个数据源,从而进行操作不同数据库中数据. 正文 JdbcTemplate 多数据源 配 ...
- 学习Spring Boot:(二十)使用 MongoDB
前言 MongoDB 1 是可以应用于各种规模的企业.各个行业以及各类应用程序的开源数据库.基于分布式文件存储的数据库.由C++语言编写.旨在为WEB应用提供可扩展的高性能数据存储解决方案.Mongo ...
- Spring Boot(十二)单元测试JUnit
一.介绍 JUnit是一款优秀的开源Java单元测试框架,也是目前使用率最高最流行的测试框架,开发工具Eclipse和IDEA对JUnit都有很好的支持,JUnit主要用于白盒测试和回归测试. 白盒测 ...
- 学习Spring Boot:(二十三)Spring Boot 中使用 Docker
前言 简单的学习下怎么在 Spring Boot 中使用 Docker 进行构建,发布一个镜像,现在我们通过远程的 docker api 构建镜像,运行容器,发布镜像等操作. 这里只介绍两种方式: 远 ...
- 学习Spring Boot:(二)启动原理
前言 主要了解前面的程序入口 @@SpringBootApplication 这个注解的结构. 正文 参考<SpringBoot揭秘 快速构建微服务体系>第三章的学习,总结下. Sprin ...
随机推荐
- 基于tensorflow的躲避障碍物的ai训练
import pygameimport randomfrom pygame.locals import *import numpy as npfrom collections import deque ...
- Spring Boot Admin 日志查看功能
按照官方配置POM和配置文件后,能够结合Eureka查看各微服务状态,但是日志始终查看不了,出现406等错误. 最后偶然发现,是在在从官方网站拷贝配置的时候,出现的问题. logging.file=* ...
- flask多app和栈的应用
一.简介 flask的蓝图可以实现url的分发,当有多个app时也可以利用app进行url分发,这里介绍下使用方式和内部原理以及栈的应用. 二.多app使用 使用示例 from werkzeu ...
- [Oracle]如何观察Table 的各种Lock 之间的冲突
[Oracle]如何观察Table 的各种Lock 之间的冲突 举例: Session#15 创建表: SID 15==============create table t1 (c1 number)p ...
- Linux系统特殊变量
系统给定的特殊变量: 变量名 作用 $0 当前脚本的名字 $n 传递给脚本或者函数的参数,n表示第几个参数 $# 传递给脚本或函数的参数个数 $* 传递给脚本或函数的所有参数 $@ 传递给脚本或者函数 ...
- Roslyn入门(一)-C#语法分析
演示环境 Visual Studio 2017 .NET Compiler Platform SDK 简介 今天,Visual Basic和C#编译器是黑盒子:输入文本然后输出字节,编译管道的中间阶段 ...
- Spring+SpringMVC+MyBatis+easyUI整合进阶篇(八)线上Mysql数据库崩溃事故的原因和处理
前文提要 承接前文<一次线上Mysql数据库崩溃事故的记录>,在文章中讲到了一次线上数据库崩溃的事件记录,建议两篇文章结合在一起看,不至于摸不着头脑. 由于时间原因,其中只讲了当时的一些经 ...
- VS2013软件的安装和单元测试
VS2013是什么? 微软在Builder 2013开发者大会上发布了Visual Studio 2013预览版,并且发布其程序组件库.NET 4.5.1的预览版.该软件已于北京时间2013年11月1 ...
- Java多线程的使用以及原理
Java有两种方式实现多线程. 第一种——继承Thread类,并重写run方法 步骤: 定义类继承Thread类: 重写子类的run方法,将线程需要执行的代码写在run方法中: 创建子类的对象,则创建 ...
- github个人作业
信息学院本科生课程设计 题目 文件加密和解密 课程名称 面向对象程序设计课程设计 课程编号 X031749 所在专业 计算机科学与技术 所在班级 计科高职13-3 ...