Spring的aop(面向切面編程)是通過(guò)代理實(shí)現(xiàn),Spring的代理分為2種岁经。
- JDK dynamic proxies(JDK)
- CGLIB
JDK
默認(rèn)情況下蝙场,當(dāng)一個(gè)類實(shí)現(xiàn)了接口,Spring 就會(huì)使用JDK代理,但是只代理接口的方法兔综。如果一個(gè)對(duì)象有其它非接口方法,非接口方法是不會(huì)被代理的槐臀。
CGLIB
CGLIB底層:使用字節(jié)碼處理框架ASM涮毫,來(lái)轉(zhuǎn)換字節(jié)碼并生成新的類蜓肆。
CGLIB代理會(huì)創(chuàng)建2個(gè)對(duì)象,一個(gè)是被代理對(duì)象舀奶,一個(gè)是實(shí)現(xiàn)了advice(增強(qiáng)邏輯,或切面邏輯)的sub class暑竟。所以CGLIB不能代理final修飾的class。
容易混淆的代理語(yǔ)義
public class SimplePojo implements Pojo {
public void foo() {
// this next method invocation is a direct call on the 'this' reference
this.bar();
}
public void bar() {
// some logic...
}
}
foo方法直接調(diào)用
public class Main {
public static void main(String[] args) {
Pojo pojo = new SimplePojo();
// this is a direct method call on the 'pojo' reference
pojo.foo();
}
}
foo方法代理調(diào)用
public class Main {
public static void main(String[] args) {
ProxyFactory factory = new ProxyFactory(new SimplePojo());
factory.addInterface(Pojo.class);
factory.addAdvice(new RetryAdvice());
Pojo pojo = (Pojo) factory.getProxy();
// this is a method call on the proxy!
pojo.foo();
}
}
foo方法是被代理的育勺,但是foo方法調(diào)用bar方法時(shí)用的是this,所以bar方法不會(huì)被代理但荤。
bar方法的代理調(diào)用
public class SimplePojo implements Pojo {
public void foo() {
// this works, but... gah!
((Pojo) AopContext.currentProxy()).bar();
}
public void bar() {
// some logic...
}
}
((Pojo) AopContext.currentProxy()).bar(); 是先獲取代理對(duì)象,再調(diào)用bar方法涧至。
ps:這里只是為了方便闡述代理才使用 ((Pojo) AopContext.currentProxy());
((Pojo) AopContext.currentProxy()); 會(huì)是代碼跟spring 強(qiáng)耦合腹躁,不建議這樣使用。
bar方法的代理調(diào)用方案思考
目前主要有幾種方式
- 將foo方法和bar方法獨(dú)立成2個(gè)對(duì)象南蓬。
- 將SimplePojo 對(duì)象再注入到 SimplePojo
SimplePojo {
SimplePojo self;
}
判斷是否被代理纺非,區(qū)分是JDK代理還是CGLIB代理
//org.springframework.aop.support.AopUtils
AopUtils.isAopProxy
AopUtils.isJdkDynamicProxy
AopUtils.isCglibProxy
AspectJ
AspectJ 沒(méi)有上面的 self-invocation 問(wèn)題,因?yàn)锳spectJ不是一個(gè)proxy的aop框架赘方。
Aspectj原理:通過(guò)asm操作Java字節(jié)碼的方式烧颖。
參考:
https://docs.spring.io/spring-framework/docs/3.0.0.M3/reference/html/ch08s06.html