-
一、前言
-
二、问题
-
三、@Inherited
-
四、@EnableAsync
-
五、解决
-
5-1、在自定义注解上加入注解 @Inherited
-
5-2、开启cglig代理
一、前言
在SpringBoot项目中,我们要开启异步执行其实很简单,只需要2步
-
在启用类上加开启异步注解 @EnableAsync
-
在对应的方法或者类上面加入标示开启异步的注解 @Async
二、问题
但是今天遇到一个问题,由于数据量的递增需把原来的方法改成异步。接到需求的时候眉头紧锁看似在思考如何处理,心里却想这难不倒我,两个注解即可完成。
但在实际操作中却出了问题,加上注解后启动项目报了空指针异常,经过定位发现错误是因为无法获取bean上自定义注解导致的。
三、@Inherited
在具体解决问题之前,我们先来了解一下这个 @Inherited
@Inherited是一个标识,用来修饰注解作用:如果一个类用上了@Inherited修饰的注解,那么其子类也会继承这个注解
注意:
-
接口用上个@Inherited修饰的注解,其实现类不会继承这个注解 -
父类的方法用了@Inherited修饰的注解,子类也不会继承这个注解 -
当用了@Inherited修饰的注解的@Retention是RetentionPolicy.RUNTIME,则增强了继承性,在反射中可以获取得到
四、@EnableAsync
在使用@EnableAsync
开启异步的时候,其实是对类生成代理来实现,而默认是使用 JDK代理
。
-
jdk代理:通过反射,对接口生成代理,生成的是和原类同级别的 兄弟关系。 -
cglib代理:通过修改字节码,生成原类的子类是 父子关系。
jdk代理的是兄弟关系会导致 @Inherited
失效
五、解决
5-1、在自定义注解上加入注解 @Inherited
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface CityMapHandleType {
CityMapTaskTypeEnum sourceType() default CityMapTaskTypeEnum.UNKNOWN;
}
5-2、开启cglig代理
@EnableAsync(proxyTargetClass = true)
proxyTargetClass 字段的含义
指示是否要创建基于子类 (CGLIB) 的代理,而不是创建标准的基于 Java 接口的代理。仅当模式设置为 AdviceMode.PROXY 时才适用。
默认值为false。
请注意,将此属性设置为 true 将影响所有需要代理的 Spring 管理的 bean,而不仅仅是标有 @Async 的那些。例如,其他标有 Spring 的 @Transactional 注解的 bean 将同时升级为子类代理。这种方法在实践中没有负面影响,除非明确期望一种类型的代理与另一种类型的代理——例如,在测试中。
原文始发于微信公众号(小道仙97):SpringBoot使用异步无法获取自定义注解问题解决
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/41308.html