[TOC]
# 1. 前置增强
(1)前置增强配置。
```xml
<aop:config>
<aop:pointcut id="add" expression="execution(public void add(int, int))" />
<aop:aspect ref="aspectLogger">
<aop:before method="printLoggerBefore" pointcut-ref="add" />
</aop:aspect>
</aop:config>
```
(2)增强函数。
```java
public void printLoggerBefore(JoinPoint jPoint) {
logger.info("\n------增强函数的输出------"
+ "\n增强方式:前置增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs()));
}
```
(3)测试结果,前置增强函数的输出要在目标函数的前面。
```
------增强函数的输出------
增强方式:前置增强
目标对象:com.learn.spring.aop.service.impl.StudentServiceImpl@515aebb0
目标函数:add
目标函数的参数:[10, 20]
------目标函数的输出------
30
```
* 前置增强先于目标函数运行。
* 无论目标函数是否发生异常,前置增强都正常执行,相当于放在`finally`语句块中执行。
* 不能获取目标函数的返回值。
# 2. 最终增强
(1)最终增强配置。
```xml
<aop:config>
<aop:pointcut id="add" expression="execution(public void add(int, int))" />
<aop:aspect ref="aspectLogger">
<aop:after method="printLoggerAfter" pointcut-ref="add" />
</aop:aspect>
</aop:config>
```
(2)增强函数。
```java
public void printLoggerAfter(JoinPoint jPoint) {
logger.info("\n------增强函数的输出------"
+ "\n增强方式:最终增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs()));
}
```
(3)测试结果,最终增强函数的输出要在目标函数的后面。
```
------目标函数的输出------
30
------增强函数的输出------
增强方式:最终增强
目标对象:com.learn.spring.aop.service.impl.StudentServiceImpl@7e07db1f
目标函数:add
目标函数的参数:[10, 20]
```
* 目标函数先于最终增强执行。
* 无论目标函数是否发生异常,最终增强都正常执行。
* 无法获取目标函数的返回值。
# 3. 异常增强
(1)异常增强配置。
```xml
<aop:config>
<aop:pointcut id="add" expression="execution(public void add(int, int))" />
<aop:aspect ref="aspectLogger">
<!-- throwing: 目标函数发生的异常,如果发生异常则由printLoggerThrowing函数捕捉 -->
<aop:after-throwing method="printLoggerThrowing" pointcut-ref="add" throwing="e" />
</aop:aspect>
</aop:config>
```
(2)增强函数。
```java
/**
* @param e e的命名与 throwing="e" 要一致
*/
public void printLoggerThrowing(JoinPoint jPoint, Exception e) {
logger.info("\n------增强函数的输出------"
+ "\n增强方式:异常增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs())
+ "\n目标函数的异常:" + e);
}
```
(3)测试结果,只有当目标函数发生异常时异常增强才执行。
```
########## 没有发生异常的结果 ###########
------目标函数的输出------
30
########## 发生异常的结果 ###########
------增强函数的输出------
增强方式:异常增强
目标对象:com.learn.spring.aop.service.impl.StudentServiceImpl@1b7cc17c
目标函数:add
目标函数的参数:[10, -20]
目标函数的异常:java.lang.Exception: (x+y)不能小于0.
java.lang.Exception: (x+y)不能小于0.
at com.learn.spring.aop.service.impl.StudentServiceImpl.add(StudentServiceImpl.java:13)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:497)
at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:344)
at org.springframework.aop.framework.ReflectiveMethodInvocation.invokeJoinpoint(ReflectiveMethodInvocation.java:198)
at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:163)
at org.springframework.aop.aspectj.AspectJAfterAdvice.invoke(AspectJAfterAdvice.java:47)
```
* 只有在目标函数发生异常时才执行。
* 目标函数先于异常增强执行。
* 不能获取目标函数的返回值。
# 4. 后置增强
(1)后置增强配置。
```xml
<aop:config>
<aop:pointcut id="sub" expression="execution(public int sub(int, int))" />
<aop:aspect ref="aspectLogger">
<!-- returning: 获取目标函数的返回值,并发送到printLoggerReturning方法中-->
<aop:after-returning method="printLoggerReturning" pointcut-ref="sub" returning="result" />
</aop:aspect>
</aop:config>
```
(2)增强函数。
```java
/**
* @param result result的命名与 returning="result" 要一致
*/
public String printLoggerReturning(JoinPoint jPoint, Object result) {
logger.info("\n------增强函数的输出------"
+ "\n增强方式:后置增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs())
+ "\n目标函数的返回值:" + result);
return "add-" + result;
}
```
(3)测试结果,可以获取目标函数的返回值。
```
------目标函数的输出------
10
------增强函数的输出------
增强方式:后置增强
目标对象:com.learn.spring.aop.service.impl.StudentServiceImpl@1283bb96
目标函数:sub
目标函数的参数:[20, 10]
目标函数的返回值:10
```
* 只有目标函数不发生异常时才执行。
* 目标函数先于后置增强执行。
* 可以获取目标函数的返回值,但不能修改目标函数的返回值,如果目标函数返回`void`,则result为`null`。
# 5. 环绕增强
(1)环绕增强配置。
```xml
<aop:config>
<aop:pointcut id="sub" expression="execution(public int sub(int, int))" />
<aop:aspect ref="aspectLogger">
<aop:around method="printLoggerAround" pointcut-ref="sub" />
</aop:aspect>
</aop:config>
```
(2)增强方法。
```java
/**
* @param jPoint ProceedingJoinPoint接口是JoinPoint的子接口,它只能用在环绕增强中
* @return 环绕增强函数返回的值将作为目标函数的返回值
*/
public int printLoggerAround(ProceedingJoinPoint jPoint) throws Throwable {
Object[] args = jPoint.getArgs();
logger.info("\n------增强函数的输出------"
+ "\n增强方式:环绕增强"
+ "\n目标对象:" + jPoint.getTarget()
+ "\n目标函数:" + jPoint.getSignature().getName()
+ "\n目标函数的参数:" + Arrays.toString(jPoint.getArgs()));
jPoint.proceed(args);
int result = ((int)args[0] - (int)args[1]) * 100;
return result;
}
```
(3)测试结果,成功修改了目标函数的返回值。
```
------增强函数的输出------
增强方式:环绕增强
目标对象:com.learn.spring.aop.service.impl.StudentServiceImpl@145f66e3
目标函数:sub
目标函数的参数:[20, 10]
------目标函数的输出------
10
返回结果:1000
```
* 环绕增强在目标函数前后都执行。
* 环绕增强函数的返回值作为目标函数的返回值,即增强方法可以改变目标函数的返回值(注意:增强并不能获取目标函数的返回值,但能修改)。
* 如果增强函数返回值为`void`,则目标函数的返回值为`null`。
- Mybatis
- mybatis是什么
- mybatis优缺点
- 环境搭建
- 使用步骤
- 传参方式
- 无需传参
- 一个参数
- 多个参数
- 增/删/改
- 查询
- 单表查询
- 一对一查询
- 一对多查询
- 动态SQL
- 注解操作
- Spring
- Spring什么
- Spring优点
- Spring组成
- 第一个Spring程序
- 两大核心技术
- IoC控制反转
- IoC思想
- IoC容器使用步骤
- 属性注入
- IoC注入方式
- 模拟IoC实现
- AOP
- AOP概念
- AOP原理
- AOP关键术语
- AOP编程过程
- 切入点规则
- 5种增强方式
- Spring注解开发
- 注解开发的优势
- Bean注解开发
- AOP注解开发
- 完全注解开发
- 模拟Spring注解开发
- 自动装配
- 配置文件拆分
- SpringBean
- Bean常用属性
- Bean的作用域
- Bean的生命周期
- Spring整合MyBatis
- 整合步骤
- SqlSessionTemplate
- 业务层添加事务
- 事务的作用
- 配置文件事务
- 注解事务
- 事务参数
- SpringMVC
- SpringMVC是什么
- 环境搭建
- 请求流程
- 核心组件
- 前后端交互
- 简单交互演示
- 常用注解
- 后端数据传递至前端
- ServletAPI
- 访问静态资源
- 异常处理
- HandlerExceptionResolver
- 局部异常
- 全局异常
- 转发与重定向
- 转发演示
- 重定向演示
- 转发与重定向的区别
- 获取表单数据
- 表单标签
- REST风格的URL
- 异步处理
- 异步请求
- JSON数据处理
- 中文乱码处理
- 日期处理
- 上传文件
- 拦截器
- 视图解析器
- 视图类型
- 多视图解析器
- 自定义pdf视图
- JSR303数据验证
- JSR303是什么
- 常用约束
- 使用步骤
- SpringMVC整合Mybatis
- 整合步骤
- Mybatis分页插件
- SpringBoot
- SpringBoot是什么
- 环境搭建
- SpringBoot启动分析
- SpringBoot启动类
- 启动过程
- SpringBoot配置文件
- 配置文件类型
- 更改配置文件
- 读取配置文件
- 占位符
- 配置优先级
- 自定义IoC容器
- 定义方式
- 引入Spring配置文件
- @Configuration
- SpringBoot自动配置
- 自动配置原理
- 条件注解
- 自动配置报告
- 自定义自动配置
- 关闭自动配置
- 接管自动配置
- 多环境配置
- CommandLineRunner
- SpringBoot与Web开发
- 引入模板引擎
- Thymeleaf模板
- Freemarker模板
- 静态资源访问
- webjars
- 静态资源位置
- ico图标
- 指定首页
- 更换Web服务器
- 国际化
- 拦截器
- 错误处理机制
- 错误处理机制原理
- 定制错误页面
- 定制错误数据
- 上传文件
- 注册servlet三大组件
- 注册Servlet
- 注册过滤器
- 注册监听器
- 外部Tomcat与jsp模板
- 前后端交互
- 传递json字符串
- 传递js对象
- 传递表单
- 下载功能
- Swagger2文档
- SpringBoot整合JDBC
- 整合步骤
- 核心API
- JdbcTemplate
- 增删改
- 查询
- NamedParameterJdbcTemplate
- 增删改
- 查询
- SpringBoot整合Mybatis
- 整合步骤
- 切换为Druid数据源
- 添加事务
- Mybatis分页插件
- 场景启动器
- 场景启动器是什么
- 自定义场景启动器
- SpringBoot与日志
- 日志框架
- slf4j日志
- slf4j日志实现
- 统一切换为slf4j
- 日志配置
- 日志文件
- 切换日志框架
- 切换日志场景启动器
- SpringBoot与缓存
- JSR107缓存技术
- Spring缓存抽象
- 缓存注解
- SpEL表达式
- 使用缓存
- 自定义key生成器
- 缓存工作原理与流程
- SpringBoot整合Redis
- 整合步骤
- 初步使用
- 序列化机制
- 缓存管理器
- SpringBoot与任务
- 异步任务
- 实现异步任务
- 注意事项与原理
- 自定义线程池
- 定时任务
- cron表达式
- 创建定时任务
- @Scheduled参数
- 动态时间
- 邮件任务
- Quartz定时任务
- Quartz是什么
- 创建定时任务
- 触发器与任务
- 任务的CURD
- 两种触发器
- 并发问题
- 持久化
- 任务持久化
- Quartz集群
- misfire策略
- 打包插件
- appassembler-maven-plugin
- appassembler与assembly配合