[TOC]
# 1. AOP原理
1. 將復雜的需求分解出不同方面,將散布在系統中的公共功能集中解決,如常用的日志、事務管理、權限管理等。
2. 采用代理機制組裝成起來運行,在不改變原程序的基礎上對代碼進行增強處理,添加新的功能。
在 Spring 中 AOP 代理使用 JDK 動態代理或 CGLIB 動態代理來實現,如果目標對象是接口,則默認采用JDK 動態代理,否則使用 CGLIB 代理。
(1)JDK動態代理
```java
// 基于接口實現的代理為JDK動態代理
public interface UserDao {...}
public class UserDaoImpl implements UserDao {...}
```
(2)CGLIB動態代理
```java
// 基于類實現的代理為CGLIB動態代理
public class People{ ... }
public Persion extends People{...}
```
<br/>
# 2. JDK動態代理演示
案例代碼:https://gitee.com/flymini/codes01/tree/master/spring_/com-learn-spring05
****
步驟如下:
**1. 創建接口**
```java
public interface UserDao {
public int add(int a,int b);
}
```
**2. 實現接口**
```java
public class UserDaoImpl implements UserDao {
@Override
public int add(int a, int b) {
return a + b;
}
}
```
**3. 創建代理對象**
```java
public class JdkDynamicProxy implements InvocationHandler {
private Object obj;
public JdkDynamicProxy(Object obj) {
this.obj = obj;
}
/**
* 增強邏輯:在invoke方法中,你可以決定在被代理對象的目標方法執行之前或之后自由決定做你想要的處理。
*
* @param proxy 需要被代理的對象
* @param method 當前被代理對象正在被調用的方法
* @param args 被代理對象方法的參數
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//方法之前
System.out.println("方法之前執行:" + method.getName() + ",傳遞的參數:" + Arrays.toString(args));
//被增強的方法執行
Object res = method.invoke(obj, args);
//方法之后
System.out.println("方法之后執行:" + obj);
return res;
}
}
```
**4. 測試**
```java
public class JdkDynamicProxyTest {
/**
* 寫法一
*/
@Test
public void proxyTest1() {
Class[] interfaces = {UserDao.class};
UserDaoImpl userDaoImpl = new UserDaoImpl();
//使用代理創建UserDao接口的一個對象
UserDao userDao = (UserDao) Proxy.newProxyInstance(JdkDynamicProxyTest.class.getClassLoader(), interfaces, new JdkDynamicProxy(userDaoImpl));
int result = userDao.add(1, 2);
System.out.println("輸出結果:" + result);
}
/**
* 寫法二
*/
@Test
public void proxyTest2() {
Class[] interfaces = {UserDao.class};
UserDaoImpl userDaoImpl = new UserDaoImpl();
UserDao userDao = (UserDao) Proxy.newProxyInstance(JdkDynamicProxyTest.class.getClassLoader(), interfaces, (proxy, method, args) -> {
//方法之前
System.out.println("方法之前執行:" + method.getName() + ",傳遞的參數:" + Arrays.toString(args));
//被增強的方法執行
Object res = method.invoke(userDaoImpl, args);
//方法之后
System.out.println("方法之后執行:" + userDaoImpl);
return res;
});
int result = userDao.add(1, 2);
System.out.println("輸出結果:" + result);
}
}
```
結果如下:
```
方法之前執行:add,傳遞的參數:[1, 2]
方法之后執行:com.learn.spring05.dao.impl.UserDaoImpl@4f8e5cde
輸出結果:3
```
執行的順序應該如下:
```
遇到語句 int result = userDao.add(1, 2); 但還沒有執行
-->
執行 invoke 方法
-->
執行 System.out.println("方法之前執行:" + method.getName() + ",傳遞的參數:" + Arrays.toString(args));
-->
這個時候才執行 userDao.add方法:Object res = method.invoke(obj, args);
-->
執行System.out.println("方法之后執行:" + obj);
```
結論:通過代理可以選擇在執行目標方法之前或之后做我們想要的邏輯處理,相當于在目標函數的前面或后面插了一腳,實現了切面編程思想。
- 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配合