案例源码:https://gitee.com/flymini/codes03/tree/master/learn-springpgm
****
[TOC]
# 1. 前置增强
**1. 前置增强配置**
```xml
<aop:config>
<aop:pointcut id="getUsername1Point" expression="execution(public void getUsername1(String))"/>
<aop:aspect ref="aspectLogger">
<aop:before method="printLogBefore" pointcut-ref="getUsername1Point"/>
</aop:aspect>
</aop:config>
```
**2. 增强函数**
```java
public void printLogBefore(JoinPoint jPoint) {
System.out.println("\n------增强函数的输出------"
+ "\n增强方式:前置增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs()));
}
```
**3. 测试结果,前置增强函数的输出要在目标函数的前面**
```
------增强函数的输出------
增强方式:前置增强
目标对象:learn.springpgm.service.impl.UserServiceImpl@565f390
目标函数:getUsername1
目标函数的参数:[zhangsan]
------目标函数的输出------
[getUsername1]:zhangsan
```
* 前置增强先于目标函数运行。
* 无论目标函数是否发生异常,前置增强都正常执行,相当于放在`finally`语句块中执行。
* 不能获取目标函数的返回值。
# 2. 最终增强
**1. 最终增强配置**
```xml
<aop:config>
<aop:pointcut id="getUsername2Point" expression="execution(public void getUsername2(String))"/>
<aop:aspect ref="aspectLogger">
<aop:after method="printLogAfter" pointcut-ref="getUsername2Point"/>
</aop:aspect>
</aop:config>
```
**2. 增强函数**
```java
public void printLogAfter(JoinPoint jPoint) {
System.out.println("------增强函数的输出------"
+ "\n增强方式:最终增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs()));
}
```
**3. 测试结果,最终增强函数的输出要在目标函数的后面**
```
------目标函数的输出------
[getUsername2]:zhangsan
------增强函数的输出------
增强方式:最终增强
目标对象:learn.springpgm.service.impl.UserServiceImpl@78452606
目标函数:getUsername2
目标函数的参数:[zhangsan]
```
* 目标函数先于最终增强执行。
* 无论目标函数是否发生异常,最终增强都正常执行。
* 无法获取目标函数的返回值。
# 3. 异常增强
**1. 异常增强配置**
```xml
<aop:config>
<aop:pointcut id="getUsername3Point" expression="execution(public void getUsername3(String))"/>
<aop:aspect ref="aspectLogger">
<!-- throwing: 目标函数发生的异常,如果发生异常则由printLogThrowing函数捕捉 -->
<aop:after-throwing method="printLogThrowing" pointcut-ref="getUsername3Point" throwing="ex"/>
</aop:aspect>
</aop:config>
```
**2. 增强函数**
```java
/**
* @param ex ex的命名与 throwing="ex" 要一致
*/
public void printLogThrowing(JoinPoint jPoint, Exception ex) {
System.out.println("------增强函数的输出------"
+ "\n增强方式:异常增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs())
+ "\n目标函数的异常:" + ex);
}
```
**3. 测试结果,只有当目标函数发生异常时异常增强才执行**
```
########## 没有发生异常的结果 ###########
------目标函数的输出------
[getUsername3]:zhangsan
########## 发生异常的结果 ###########
------增强函数的输出------
增强方式:异常增强
目标对象:learn.springpgm.service.impl.UserServiceImpl@c667f46
目标函数:getUsername3
目标函数的参数:[null]
目标函数的异常:java.lang.RuntimeException: 用户名不能为空
java.lang.RuntimeException: 用户名不能为空
at learn.springpgm.service.impl.UserServiceImpl.getUsername3(UserServiceImpl.java:30)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:77)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:568)
at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:343)
```
* 只有在目标函数发生异常时才执行。
* 目标函数先于异常增强执行。
* 不能获取目标函数的返回值。
# 4. 后置增强
**1. 后置增强配置**
```xml
<aop:config>
<aop:pointcut id="getUsername4Point" expression="execution(public String getUsername4(String))" />
<aop:aspect ref="aspectLogger">
<!-- returning: 获取目标函数的返回值,并发送到printLoReturning方法中-->
<aop:after-returning method="printLogReturning" pointcut-ref="getUsername4Point" returning="result" />
</aop:aspect>
</aop:config>
```
**2. 增强函数**
```java
/**
* @param result result的命名与 returning="result" 要一致
*/
public String printLogReturning(JoinPoint jPoint, Object result) {
System.out.println("------增强函数的输出------"
+ "\n增强方式:后置增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs())
+ "\n目标函数的返回值:" + result);
return "add-" + result;
}
```
**3. 测试结果,可以获取目标函数的返回值**
```
------目标函数的输出------
[getUsername4]:zhangsan
------增强函数的输出------
增强方式:后置增强
目标对象:learn.springpgm.service.impl.UserServiceImpl@19976a65
目标函数:getUsername4
目标函数的参数:[zhangsan]
目标函数的返回值:[getUsername4]:zhangsan
```
* 只有目标函数不发生异常时才执行。
* 目标函数先于后置增强执行。
* 可以获取目标函数的返回值,但不能修改目标函数的返回值,如果目标函数返回`void`,则 result 为`null`。
# 5. 环绕增强
**1. 环绕增强配置**
```xml
<aop:config>
<aop:pointcut id="getUsername5Point" expression="execution(public String getUsername5(String))"/>
<aop:aspect ref="aspectLogger">
<aop:around method="printLogAround" pointcut-ref="getUsername5Point"/>
</aop:aspect>
</aop:config>
```
**2. 增强方法**
```java
/**
* @param jPoint ProceedingJoinPoint接口是JoinPoint的子接口,它只能用在环绕增强中
* @return 环绕增强函数返回的值将作为目标函数的返回值
*/
public String printLogAround(ProceedingJoinPoint jPoint) throws Throwable {
Object[] args = jPoint.getArgs();
System.out.println("------增强函数的输出------"
+ "\n增强方式:环绕增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs()));
jPoint.proceed(args);
return "[printLogAround]-" + args[0];
}
```
**3. 测试结果,成功修改了目标函数的返回值**
```
------增强函数的输出------
增强方式:环绕增强
目标对象:learn.springpgm.service.impl.UserServiceImpl@a1f72f5
目标函数:getUsername5
目标函数的参数:[zhangsan]
------目标函数的输出------
本应该返回值为:[getUsername5]:zhangsan
增强后返回值为:[printLogAround]-zhangsan
```
* 环绕增强在目标函数前后都执行。
* 环绕增强函数的返回值作为目标函数的返回值,即增强方法可以改变目标函数的返回值(注意:增强并不能获取目标函数的返回值,但能修改)。
* 如果增强函数返回值为`void`,则目标函数的返回值为`null`。
- Spring
- Spring是什么
- Spring与EJB对比
- Spring的组成
- 首个Spring程序
- IoC控制反转
- 什么是IoC
- IoC编程
- 依赖注入方式
- 不同变量注入
- AOP面向切面编程
- AOP思想
- AOP实现原理
- AOP关键术语
- AOP编程
- 5种增强方式
- 切入点规则
- 自动装配
- Spring注解开发
- Bean注解
- AOP注解
- 完全注解
- 配置文件拆分
- SpringBean
- Bean常用属性
- Bean作用域
- Bean生命周期
- SpringBoot
- SpringBoot是什么
- 项目创建
- 配置文件
- 配置类型
- 读取配置
- 占位符
- 多环境配置
- 配置优先级
- 更改配置文件
- 自定义IoC容器
- 常用组件
- ApplicationContextAware
- CommandLineRunner
- Boot[Web]
- 引入模板引擎
- 静态资源访问
- 指定首页
- JSP支持
- 注册拦截器
- 注册Servlet组件
- 注册Servlet
- 注册过滤器
- 注册监听器
- 拦截器与过滤器区别
- 文件上传
- 文件下载
- 变更服务器
- Controller层封装
- HttpServletRequest
- 获取请求行
- 获取请求头
- 获取请求体
- Boot[自动配置]
- 自动配置是什么
- 自动配置报告
- 关闭自动配置
- 条件注解
- Boot[场景启动器]
- 场景启动器是什么
- 自定义场景启动器
- Boot[日志]
- 日志框架
- 日志级别
- 日志配置
- 配置文件
- 切换日志
- Boot[邮件任务]
- Boot[定时任务]
- cron表达式
- 起步
- 任务并行
- 注解Scheduled参数
- Boot[异步任务]
- 起步
- 注意事项与原理
- 自定义线程池
- Boot[缓存]
- JSR107缓存技术
- Spring缓存抽象
- 缓存注解
- SpEL表达式
- 起步
- 自定义key生成器
- 工作原理
- Boot[Redis]
- 起步
- 序列化机制
- Boot[Jdbc]
- 起步
- 两个模板类
- JdbcTemplate
- 增删改
- 查询
- NamedParameterJdbcTemplate
- 增删改
- 查询
- 自定义JdbcTemplate
- Boot[JPA]
- SpringDataJPA是什么
- 与JPA、Hibernate的关系
- 起步
- SpringDataJPA原理
- 查询方式
- 方法命名规则查询
- 限制查询结果查询
- 注解Query查询
- 命名参数查询
- SpEL表达式查询
- 原生查询
- 更新与删除
- 查询指定字段
- Specification动态查询
- 分页查询与排序
- 多表查询
- 一对一查询
- 一对多查询
- 多对多查询
- Specification查询
- Query注解查询
- 主键策略
- 单独主键
- 联合主键
- 级联操作
- 加载规则
- 审计功能
- 常用注解
- 避坑指南
- Boot[JSR303]
- JSR303是什么
- 常用约束
- 起步
- 简单校验
- 嵌套校验
- 分组校验
- 自定义约束注解
- 自定义校验工具
- Spring事务
- 事务的作用
- 起步
- 事务参数
- SpringDoc文档
- SpringDoc是什么
- 起步
- 自定义配置
- 常用Doc注解
- JSR303文档
- knife4j文档
- 常用配置
- Boot[RabbitMQ]
- 起步
- Fanout交换机类型
- Direct交换机类型
- Topic交换机类型
- 延迟队列插件
- RabbitListener监听方法
- JWT认证
- 认证流程
- 起步
- 密码加密
- JWT认证实现