SpringAop實現(xiàn)原理及代理模式詳解
Spring Aop的原理
Spring的AOP就是通過動態(tài)代理實現(xiàn)的。當為某個Bean或者某些Bean配置切面時,Spring會為其創(chuàng)建代理對象,當調(diào)用該對象的某個方法時,實際是調(diào)用生成的代理類的對象方法。Spring的Aop主要是使用了兩個動態(tài)代理,分別是JDK的動態(tài)代理和CGLIB動態(tài)代理。
1. JDK動態(tài)代理
如果代理類實現(xiàn)了接口,Spring默認會使用JDK動態(tài)代理。JDK的動態(tài)代理是基于反射實現(xiàn)。JDK通過反射,生成一個代理類,這個代理類實現(xiàn)了原來那個類的全部接口,并對接口中定義的所有方法進行了代理。當我們通過代理對象執(zhí)行原來那個類的方法時,代理類底層會通過反射機制,調(diào)用我們實現(xiàn)的InvocationHandler接口的invoke方法。
/*
* 接口類
*/
public interface Person {
void say();
}
* 接口實現(xiàn)類
public class Man implements Person {
private String word;
public Man(String word){
this.word = word;
}
public Man(){
public void say(){
System.out.println("Man Can Say " + word);
public class ManJDKProxy implements InvocationHandler {
/**
* 需要的代理對象
*/
private Object o;
public Object bind(Object o){
this.o = o;
return Proxy.newProxyInstance(o.getClass().getClassLoader(), o.getClass().getInterfaces(), this);
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("JDK Proxy Design");
return method.invoke(o, args);
/**
* JDK動態(tài)代理
public class ProxyDesign_2 {
public static void main(String[] args) {
Man man = new Man("Hello");
Person p = (Person)new ManJDKProxy().bind(man);
p.say();* JDK動態(tài)代理的優(yōu)缺點
優(yōu)點:
1. JDK動態(tài)代理是JDK原生的,不需要任何依賴即可使用
2. 通過反射機制生成代理類的速度要比CGLib操作字節(jié)碼生成代理類的速度更快
缺點:
1. 如果要使用JDK動態(tài)代理,被代理的類必須實現(xiàn)了接口,否則無法代理(InvocationHandler)
2. JDK動態(tài)代理無法為沒有在接口中定義的方法實現(xiàn)代理
3. JDK動態(tài)代理執(zhí)行代理方法時,需要通過反射機制進行回調(diào),此時方法執(zhí)行的效率比較低
2. CGLIB動態(tài)代理
若需要代理的類沒有實現(xiàn)接口,JDK的動態(tài)代理就無法使用,Spring會使用CGLiB動態(tài)代理來生成代理對象。CGLiB直接操作字節(jié)碼,生成類的子類,重寫類的方法完成代理。
/*
* 接口類
*/
public interface Person {
void say();
}
* 接口實現(xiàn)類
public class Man implements Person {
private String word;
public Man(String word){
this.word = word;
}
public Man(){
public void say(){
System.out.println("Man Can Say " + word);
public class ManCGLIBProxy {
public Object bind(Object target){
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
enhancer.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("CGLIB Proxy Design");
return method.invoke(target, objects);
}
});
return enhancer.create();
/**
* CGLIB動態(tài)代理
public class ProxyDesign_3 {
public static void main(String[] args) {
Man man = new Man("Hello");
Person p = (Person)new ManCGLIBProxy().bind(man);
p.say();* CGLiB動態(tài)代理的優(yōu)缺點
優(yōu)點:
1. 使用CGLiB代理的類,不需要實現(xiàn)接口,因為CGLib生成的代理類是直接繼承自需要被代理的類
2. 因為CGLiB實現(xiàn)方式是重寫父類的方法,所以對final方法,或者private方法是沒有辦法代理的
3. CGLiB是通過修改字節(jié)碼生成的代理類,所以CGLib執(zhí)行代理方法的效率要高于JDK的動態(tài)代理
缺點:
1. 因為CGLiB實現(xiàn)方式是重寫父類的方法,所以對final方法,或者private方法是沒有辦法代理的
2. 因為CGLiB生成代理類的方式是通過操作字節(jié)碼(asm工具包),這種生成的代理類的方式比JDK通過反射生成代理類的方式的效率低
3. Spring項目中如何強制使用CGLIB代理方式
* xml方式
<!-- aop:config用來在xml中配置切面,指定proxy-target-class="true" --> <aop:config proxy-target-class="true"> <!-- AOP相關(guān)配置 --> </aop:config>
* @Aspect注解方式
<!-- 將proxy-target-class配置設(shè)置為true --> <aop:aspectj-autoproxy proxy-target-class="true"/>
* 配置類注解方式
添加@EnableAspectJAutoProxy(proxyTargetClass = true)
到此這篇關(guān)于SpringAop實現(xiàn)原理及代理模式的文章就介紹到這了,更多相關(guān)SpringAop代理模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決@PathVariable參數(shù)接收不完整的問題
這篇文章主要介紹了解決@PathVariable參數(shù)接收不完整的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
Java結(jié)構(gòu)型設(shè)計模式中代理模式示例詳解
代理模式(Proxy Parttern)為一個對象提供一個替身,來控制這個對象的訪問,即通過代理對象來訪問目標對象。本文將通過示例詳細講解一下這個模式,需要的可以參考一下2022-09-09
springboot編程式事務(wù)TransactionTemplate的使用說明
這篇文章主要介紹了springboot編程式事務(wù)TransactionTemplate的使用說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06
mybatis Example Criteria like 模糊查詢問題
這篇文章主要介紹了mybatis Example Criteria like 模糊查詢問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-02-02
SpringBoot實現(xiàn)識別圖片中的身份證號與營業(yè)執(zhí)照信息
這篇文章主要為大家詳細介紹了SpringBoot如何實現(xiàn)識別圖片中的身份證號與營業(yè)執(zhí)照信息,文中的示例代碼講解詳細,感興趣的小伙伴可以了解一下2024-01-01

