[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配合