Spring在實現AOP時需要用到代理,有兩種實現可選的代理方式,分別是JDK動態代理以及CGLIB代理。但這兩種代理卻又有着很大的區別。
一、兩種實現代理方式的介紹
- JDK動態代理
源自Java純正血統,屬於Java SE規範中的API。實現動態代理有如下四步驟:
- 通過實現 InvocationHandler 接口創建自己的調用處理器;
- 通過爲 Proxy 類指定 ClassLoader 對象和一組 interface 來創建動態代理類;
- 通過反射機制獲得動態代理類的構造函數,其唯一參數類型是調用處理器接口類型;
- 通過構造函數創建動態代理類實例,構造時調用處理器對象作爲參數被傳入。
- GCLIB代理
-
- cglib(Code Generation Library)是一個強大的、高性能、高質量的Code生成類庫。它可以在運行期擴展Java類與實現Java接口。
- cglib封裝了asm,可以在運行期動態生成新的class。
- cglib用於AOP,JDK中的proxy必須基於接口,cglib卻沒有這個限制。
二、兩種實現方式在Spring應用中的區別
- 如果目標對象實現了接口,默認情況下會採用JDK動態代理實現AOP
- 如果目標對象實現了接口,可以強制使用CGLIB實現AOP
- 如果目標對象沒有實現了接口,必須採用CGLIB庫,spring會自動在JDK動態代理和CGLIB之間轉換
- JDK動態代理只能對實現了接口的類生成代理,而不能針對類。它是利用反射機制生成一個實現代理接口的匿名類,在調用具體方法前調用InvokeHandler來處理。
- CGLIB是針對類實現代理,主要是對指定的類生成一個子類,覆蓋其中的方法,因爲是繼承,所以該類或方法最好不要聲明成final。cglib動態代理是利用asm開源包,對代理對象類的class文件加載進來,通過修改其字節碼生成子類來處理。
- JDK動態代理生成的代理類並不是被代理類的子類,它只是實現了被代理類對應接口的一個匿名類,所以接口中沒有的方法是不可用的,而且代理出來的類無法用被代理類的變量接收,但CGLIB確是被代理類的子類,所以被代理類的變量能夠接收代理出來的類。
三、如何強制使用CGLIB實現AOP
Spring在實現AOP時需要用到代理,但默認情況下,如果目標對象實現了接口,會採用JDK動態代理來實現。我們需要用以下兩部來強制使用CGLIB。
- 添加CGLIB庫,SPRING_HOME/cglib/*.jar
- 在spring配置文件中加入<aop:aspectj-autoproxy proxy-target-class="true"/>