[TOC]
# 1. 异步方法内依然是同步的
异步方法的方法内依然是同步的,方法外才是异步的。如下示例。
**1. 异步方法**
```java
@Slf4j
@Service
public class AsyncService {
@Async
public void asyncHandle1() {
String name = "zhangsan";
log.info("[asyncHandle1|name1]: {}", name);
try {
Thread.sleep(5000);
name = "lisi";
} catch (InterruptedException e) {
e.printStackTrace();
}
log.info("[asyncHandle1|name2]: {}", name);
}
}
```
**2. 其它类中调用异步方法**
```java
@Slf4j
@RestController
@RequiredArgsConstructor
public class AsyncController {
final AsyncService asyncService;
@GetMapping("/v1/asyncHandle1")
public void asyncHandle1() {
log.info("[asyncHandle1|start]");
//调用异步方法
asyncService.asyncHandle1();
log.info("[asyncHandle1|end]");
}
}
```
```
///日志打印顺序如下,异步方法的方法内依然是同步的,方法外才是异步的
2023-06-02 17:34:10 ...: [asyncHandle1|start]
2023-06-02 17:34:10 ...: [asyncHandle1|end]
2023-06-02 17:34:10 ...: [asyncHandle1|name1]: zhangsan
-- 等待 5s 后输出 lisi,而不是zhangsan
2023-06-02 17:34:15 ...: [asyncHandle1|name2]: lisi
```
<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. 异步方法返回值
可以将 Future 作为异步方法的返回值,但当调用`Future.get`时主线程就会被阻塞,这时异步方法就不再是异步执行了。
**1. 异步方法返回 Future**
```java
@Slf4j
@Service
public class AsyncService {
@Async
public Future<String> asyncHandle2() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
log.info("[asyncHandle2]");
return CompletableFuture.completedFuture("张三");
}
}
```
<br/>
**2. 其它类中调用异步方法**
```java
@Slf4j
@RestController
@RequiredArgsConstructor
public class AsyncController {
final AsyncService asyncService;
@GetMapping("/v1/asyncHandle2")
public void asyncHandle2() throws ExecutionException, InterruptedException {
log.info("[asyncHandle2|start]");
//调用异步方法
Future<String> result = asyncService.asyncHandle2();
//如果调用 get,则 asyncService.asyncHandle2()不是异步执行了
log.info("[asyncHandle2|name]: {}", result.get());
}
}
```
日志打印顺序如下,说明异步没有执行。
```
///调用 get 日志打印顺序如下,说明异步没有执行
2023-06-02 18:20:49 ...: [asyncHandle2|start]
2023-06-02 18:20:54 ...: [asyncHandle2]
-- 等待 5s 后输出
2023-06-02 18:20:54 ...: [asyncHandle2|name]: 张三
///没有调用 get 日志打印顺序如下,说明异步执行了
2023-06-02T18:23:44 ...: [asyncHandle2|start]
2023-06-02T18:23:44 ...: [asyncHandle2|name]:
-- 等待 5s 后输出
2023-06-02T18:23:49 ...: [asyncHandle2]
```
>[info]异步方法只能返回 Future,或 void。
- Spring
- Spring是什么
- Spring与EJB对比
- Spring的组成
- 首个Spring程序
- IoC控制反转
- 什么是IoC
- IoC编程
- 依赖注入方式
- 不同变量注入
- AOP面向切面编程
- AOP思想
- AOP实现原理
- AOP关键术语
- AOP编程
- 5种增强方式
- 切入点规则
- 自动装配
- Spring注解开发
- Bean注解
- AOP注解
- 完全注解
- 配置文件拆分
- SpringBean
- Bean常用属性
- Bean作用域
- Bean生命周期
- SpringBoot
- SpringBoot是什么
- 项目创建
- 配置文件
- 配置类型
- 读取配置
- 占位符
- 多环境配置
- 配置优先级
- 更改配置文件
- 自定义IoC容器
- 常用组件
- ApplicationContextAware
- CommandLineRunner
- Boot[Web]
- 引入模板引擎
- 静态资源访问
- 指定首页
- JSP支持
- 注册拦截器
- 注册Servlet组件
- 注册Servlet
- 注册过滤器
- 注册监听器
- 拦截器与过滤器区别
- 文件上传
- 文件下载
- 变更服务器
- Controller层封装
- HttpServletRequest
- 获取请求行
- 获取请求头
- 获取请求体
- Boot[自动配置]
- 自动配置是什么
- 自动配置报告
- 关闭自动配置
- 条件注解
- Boot[场景启动器]
- 场景启动器是什么
- 自定义场景启动器
- Boot[日志]
- 日志框架
- 日志级别
- 日志配置
- 配置文件
- 切换日志
- Boot[邮件任务]
- Boot[定时任务]
- cron表达式
- 起步
- 任务并行
- 注解Scheduled参数
- Boot[异步任务]
- 起步
- 注意事项与原理
- 自定义线程池
- Boot[缓存]
- JSR107缓存技术
- Spring缓存抽象
- 缓存注解
- SpEL表达式
- 起步
- 自定义key生成器
- 工作原理
- Boot[Redis]
- 起步
- 序列化机制
- Boot[Jdbc]
- 起步
- 两个模板类
- JdbcTemplate
- 增删改
- 查询
- NamedParameterJdbcTemplate
- 增删改
- 查询
- 自定义JdbcTemplate
- Boot[JPA]
- SpringDataJPA是什么
- 与JPA、Hibernate的关系
- 起步
- SpringDataJPA原理
- 查询方式
- 方法命名规则查询
- 限制查询结果查询
- 注解Query查询
- 命名参数查询
- SpEL表达式查询
- 原生查询
- 更新与删除
- 查询指定字段
- Specification动态查询
- 分页查询与排序
- 多表查询
- 一对一查询
- 一对多查询
- 多对多查询
- Specification查询
- Query注解查询
- 主键策略
- 单独主键
- 联合主键
- 级联操作
- 加载规则
- 审计功能
- 常用注解
- 避坑指南
- Boot[JSR303]
- JSR303是什么
- 常用约束
- 起步
- 简单校验
- 嵌套校验
- 分组校验
- 自定义约束注解
- 自定义校验工具
- Spring事务
- 事务的作用
- 起步
- 事务参数
- SpringDoc文档
- SpringDoc是什么
- 起步
- 自定义配置
- 常用Doc注解
- JSR303文档
- knife4j文档
- 常用配置
- Boot[RabbitMQ]
- 起步
- Fanout交换机类型
- Direct交换机类型
- Topic交换机类型
- 延迟队列插件
- RabbitListener监听方法
- JWT认证
- 认证流程
- 起步
- 密码加密
- JWT认证实现