大家應(yīng)該都了解Spring AOP有兩種實(shí)現(xiàn)方式鸵赖,Jdk Proxy和Cglib净当。默認(rèn)情況下齐媒,如果類實(shí)現(xiàn)了接口,則用JDK動(dòng)態(tài)代理忌卤;如果類沒有實(shí)現(xiàn)接口,則用Cglib進(jìn)行代理楞泼。z
具體實(shí)現(xiàn)代碼在DefaultAopProxyFactory.class里:
@Override
public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
Class<?> targetClass = config.getTargetClass();
if (targetClass == null) {
throw new AopConfigException("TargetSource cannot determine target class: " +
"Either an interface or a target is required for proxy creation.");
}
if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) {
return new JdkDynamicAopProxy(config);
}
return new ObjenesisCglibAopProxy(config);
}
else {
return new JdkDynamicAopProxy(config);
}
}
根據(jù)上面那個(gè)規(guī)則可能會(huì)有個(gè)問題驰徊,例如:
public interface Hello {
void say();
}
@Component
public class HelloImpl implements Hello {
@Cacheable
@Override
public void say() {
System.out.println("say time : " + System.currentTimeMillis());
}
}
@Component
public class Person {
@Autowired
private HelloImpl hello;
}
@EnableCaching
@Configuration
public class HelloConfiguration {
}
//運(yùn)行結(jié)果:
Caused by: org.springframework.beans.factory.BeanNotOfRequiredTypeException: Bean named 'helloImpl' is expected to be of type 'com.alibaba.middleware.demo.HelloImpl' but was actually of type 'com.sun.proxy.$Proxy125'
HelloImpl.say()方法用了@Cacheable,會(huì)通過AOP對(duì)HelloImpl進(jìn)行代理堕阔,對(duì)say方法進(jìn)行增強(qiáng)棍厂。這次AOP判斷HelloImpl實(shí)現(xiàn)接口Hello,所以會(huì)用JDK Proxy生成一個(gè)代理類$Proxy125超陆,$Proxy125實(shí)現(xiàn)了Hello牺弹。
Person里通過@Autowired注解注入了HelloImpl,BeanFactory會(huì)根據(jù)注入bean類型去找同樣類型的bean时呀,根據(jù)查找規(guī)則
找到了$Proxy125张漂,然后通過賦值給HelloImpl hello = $Proxy125發(fā)生了類型轉(zhuǎn)換錯(cuò)誤。
問題的原因
直接原因是谨娜,代理出來的對(duì)象$Proxy125只能轉(zhuǎn)成它的實(shí)現(xiàn)接口的類型航攒,HelloImpl不是它的接口。
深層次原因是趴梢,AOP在選擇代理類型的時(shí)候出現(xiàn)了失誤
漠畜,根據(jù)Autowired的類型币他,這個(gè)地方使用Cglib進(jìn)行代理。Cglib是通過生成代類子類的方式盆驹,這里生成HelloImpl的子類圆丹,也能完美轉(zhuǎn)成HelloImpl。
處理方式
處理辦法有幾種:
- 不注入HelloImpl躯喇,改成Hello辫封。不現(xiàn)實(shí)。廉丽。
- 指定代理類型為Cglib
指定AOP類型的方式
- 在Bean聲明是通過@Scope(proxyMode = ScopedProxyMode.TARGET_CLASS)指定Bean的代理類型為Cglib倦微。這種方式有個(gè)限制,需要有修改Bean類的權(quán)限正压。
- 指定所有Bean的代理方式為Cglib欣福。
- 通過修改BeanDefinition,設(shè)置Bean的代理類型為Cglib焦履。
2具體實(shí)現(xiàn)拓劝,xml方式
<aop:aspectj-autoproxy proxy-target-class="true"/>
@EnableAspectJAutoProxy(proxyTargetClass=true)
3具體實(shí)現(xiàn),在BeanFactoryPostProcessor對(duì)BeanDefinition進(jìn)行設(shè)置嘉裤,之后在Bean實(shí)例化的時(shí)候就會(huì)根據(jù)這個(gè)屬性選擇Cglib代理方式
BeanDefinition beanDefinition = beanFactory.getBeanDefinition(beanName);
beanDefinition.setAttribute(AutoProxyUtils.PRESERVE_TARGET_CLASS_ATTRIBUTE, Boolean.TRUE);