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。


我的头条:
在这里插入图片描述

  • 7
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值