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