本文主要是介绍Springboot 动态代理异常-because it is a JDK dynamic proxy that implements,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
用心编码:
雷军: 用互联网的方式做企业,要靠 “专注、极致、口碑、快” 这 “七字诀”!
今天在重构一套原有系统时,项目启动时,莫名其妙的报错了,错误如下图所示:
一、问题描述
***************************
APPLICATION FAILED TO START
***************************Description:The bean 'attachCacheService' could not be injected as a 'com.abc.service.AttachCacheService' because it is a JDK dynamic proxy that implements:Action:Consider injecting the bean as one of its interfaces or forcing the use of CGLib-based proxies by setting proxyTargetClass=true on @EnableAsync and/or @EnableCaching.Process finished with ex 类it code 1
错误的意思是:AttachCacheService 类采用的是 JDK动态代理的方式实现的,需要我们考虑使用基于CGLib方式实现动态代理。
二、代码分析
1.服务层(service)没有按照传统方式定义, 没有定义接口及实现。而是直接定义实现类,如下:
@Service
public class AttachCacheService{@Transactional(rollbackFor = Exception.class)public String upload(MultipartFile file) {// upload file...return "http://test.com/abcde.jpg";}
}
2.在网上查询这个错误时,我得到以下几种解决方案:
(1)在启动类上加上 @EnableTransactionManagement(proxyTargetClass = true)
(2)在配置文件配置 spring.aop.proxy-target-class=true
(3)按照传统方式定义 接口及实现类
三、深入理解Springboot 动态代理及CGLib
1.JDK动态代理
JDK动态代理:利用Java反射机制生成一个实现代理接口的匿名类,通过InvokeHandler来处理。
针对接口类生成代理。
2.CGLib动态代理
CGLib动态代理:利用开源asm包对代理对象类(class)文件动态加载,并且修改字节码数据实现动态代理。
针对类生成实现代理。
(1)若目标对象实现了接口(IService + ServiceImpl) ,那么Springboot 会采用JDK动态代理实现AOP
(2)如果目标对象没有实现接口(Service),那么必须采用CGLib 实现AOP
四、总结
1.重点是上面方法上加上了@Transactional(rollbackFor = Exception.class)注解,该注解的目标主要是用于控制数据库事务,那么它的内部必须通过AOP切面的方法去实现事务控制,所以会用到动态代理,而且是采用CGLib。
2.一般情况下 实现接口类,接口类采用JDK代理,实现类采用CGLib代理
3.Springboot2.x默认使用CGLib代理,官方如下解释:
This was changed in 1.4 (see 5423). We’ve generally found cglib proxies less likely to cause unexpected cast exceptions.
需要将 spring.aop.proxy-target-class属性值设置为true 调用cglib
4.当设置过 @EnableAsync和@EnableCaching 时,spring.aop.proxy-target-class 默认为false。
我的头条:
这篇关于Springboot 动态代理异常-because it is a JDK dynamic proxy that implements的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!