[TOC]
# 1. 異步方法內依然是同步的
異步方法的方法內依然是同步的,方法外才是異步的。如下示例。
**1. 異步方法**
```java
@Slf4j
@Service
public class AsyncService {
@Async
public void async02() {
String name = "zhangsan";
log.info("AsyncService -> async02 -> name001({})", name);
try {
Thread.sleep(5000);
name = "wangwu";
log.info("AsyncService -> async02 -> name002({})", name);
} catch (InterruptedException e) {
e.printStackTrace();
}
log.info("AsyncService -> async02 -> name003({})", name);
}
}
```
**2. 同步方法調用異步方法**
```java
@Slf4j
@RestController
public class AsyncController {
@Autowired
AsyncService asyncService;
@GetMapping("/sync02")
public void sync02() {
log.info("AsyncController -> sync02 -> 001");
//同步方法中調用異步方法
asyncService.async02();
log.info("AsyncController -> sync02 -> 002");
}
}
```
日志打印順序如下。
```
2022-09-07 11:08:36 : AsyncController -> sync02 -> 001
2022-09-07 11:08:36 : AsyncController -> sync02 -> 002
2022-09-07 11:08:36 : AsyncService -> async02 -> name001(zhangsan)
-- 等待5s輸出下面內容
2022-09-07 11:08:41 : AsyncService -> async02 -> name002(wangwu)
2022-09-07 11:08:41 : AsyncService -> async02 -> name003(wangwu)
```
<br/>
# 2. 引起@Async失效的情況
轉載自:https://zhuanlan.zhihu.com/p/462544527
****
* 異步方法使用static修飾。
* 異步類沒有使用@Component注解(或其他注解)導致spring無法掃描到異步類。
* 異步方法被同類的方法(異步/同步)調用會失效。
* 類中需要使用@Autowired或@Resource等注解自動注入,不能自己手動new對象。
* 如果使用SpringBoot框架必須在啟動類中增加@EnableAsync注解。
* 異步方法將Future作為返回值。
<br/>
# 3. @Async的原理簡述
轉載自:https://blog.csdn.net/qq_44750696/article/details/123960134
****
@Async 的原理是通過 Spring AOP 動態代理的方式來實現的。
<br/>
Spring 容器啟動初始化 bean 時,判斷類中是否使用了 @Async 注解,如果使用了則為其創建切入點和切入點處理器,根據切入點創建代理。
<br/>
在線程調用 @Async 注解標注的方法時,會調用代理,執行切入點處理器`invoke`方法,將方法的執行提交給線程池中的另外一個線程來處理,從而實現了異步執行。
<br/>
所以,需要注意的一個錯誤用法是,如果 A 方法(不論是同步還是異步)調用它同類中標注 @Async 的 B 方法,是不會異步執行的,因為從 A 方法進入調用的都是該類對象本身,不會進入代理類。因此,<mark>相同類中的方法調用帶 @Async 的方法是無法異步的,這種情況仍然是同步</mark>。
<br/>
# 4. 異步方法返回值
**1. 調用異步方法是獲取不到該異步方法的返回值的**
```java
// 異步方法
@Async
public String async03() {
String name = "zhangsan";
log.info("AsyncService -> async03 -> name001({})", name);
try {
Thread.sleep(5000);
name = "wangwu";
log.info("AsyncService -> async03 -> name002({})", name);
} catch (InterruptedException e) {
e.printStackTrace();
}
log.info("AsyncService -> async03 -> name003({})", name);
return name;
}
// 調用異步方法
@GetMapping("/sync03")
public String sync03() {
log.info("AsyncController -> sync03 -> 001");
//同步方法中調用異步方法
String name = asyncService.async03();
log.info("AsyncController -> sync03 -> name({})", name);
return name;
}
```
日志打印順序如下:
```
2022-09-07 12:03:33 : AsyncController -> sync03 -> 001
2022-09-07 12:03:33 : AsyncController -> sync03 -> name(null)
2022-09-07 12:03:33 : AsyncService -> async03 -> name001(zhangsan)
-- 等待5s打印如下內容
2022-09-07 12:03:38 : AsyncService -> async03 -> name002(wangwu)
2022-09-07 12:03:38 : AsyncService -> async03 -> name003(wangwu)
```
<br/>
**2. 如果非要獲取異步方法的返回值,可以返回Future類,但是這個方法就不再是異步方法了**
```java
// 異步方法,返回值為Future
@Async
public Future<String> async04() {
String name = "zhangsan";
log.info("AsyncService -> async04 -> name001({})", name);
try {
Thread.sleep(5000);
name = "wangwu";
log.info("AsyncService -> async04 -> name002({})", name);
} catch (InterruptedException e) {
e.printStackTrace();
}
log.info("AsyncService -> async04 -> name003({})", name);
return new AsyncResult<>(name);
}
// 調用異步方法
@GetMapping("/sync04")
public String sync04() throws ExecutionException, InterruptedException {
log.info("AsyncController -> sync04 -> 001");
//同步方法中調用異步方法
String name = asyncService.async04().get();
log.info("AsyncController -> sync04 -> name({})", name);
return name;
}
```
日志打印順序如下,說明異步沒有執行。
```
2022-09-07 12:07:08 : AsyncController -> sync04 -> 001
2022-09-07 12:07:08 : AsyncService -> async04 -> name001(zhangsan)
2022-09-07 12:07:13 : AsyncService -> async04 -> name002(wangwu)
2022-09-07 12:07:13 : AsyncService -> async04 -> name003(wangwu)
2022-09-07 12:07:13 : AsyncController -> sync04 -> name(wangwu)
```
>[warning] 1. 雖然不能獲取異步方法的返回值,但是異步方法的返回值可以不是void。<br/> 2. 只能返回引用類型,不能返回基本數據類型。
- 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配合