[TOC]
靜態代理和動態代理模式都是要求目標對象是實現一個接口的目標對象,但是有時候目標對象只是一個單獨的對象,并沒有實現任何的接口,這個時候就可以使用以目標對象子類的方式類實現代理,這種方法就叫做:Cglib代理
**Cglib代理,也叫作子類代理,它是在內存中構建一個子類對象從而實現對目標對象功能的擴展.**
* JDK的動態代理有一個限制,就是使用動態代理的對象必須實現一個或多個接口,如果想代理沒有實現接口的類,就可以使用Cglib實現.
* Cglib是一個強大的高性能的代碼生成包,它可以在運行期擴展java類與實現java接口.它廣泛的被許多AOP的框架使用,例如Spring AOP和synaop,為他們提供方法的interception(攔截)
* Cglib包的底層是通過使用一個小而塊的字節碼處理框架ASM來轉換字節碼并生成新的類.不鼓勵直接使用ASM,因為它要求你必須對JVM內部結構包括class文件的格式和指令集都很熟悉
**Cglib子類代理實現方法:**
1.需要引入cglib的jar文件,但是Spring的核心包中已經包括了Cglib功能,所以直接引入`pring-core-3.2.5.jar`即可.
2.引入功能包后,就可以在內存中動態構建子類
3.代理的類不能為final,否則報錯
4.目標對象的方法如果為final/static,那么就不會被攔截,即不會執行目標對象額外的業務方法.
5.如果方法為static,private則無法進行代理。
### Cglib代理工廠:
CglibProxyFactory.java
~~~
package com.yc.model.ProxyFactory;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
/*
Cglib代理工廠
對userDao在內存中動態構建一個子類對象
*/
public class CglibProxyFacory implements MethodInterceptor {
//維護目標對象
private Object target;
public CglibProxyFacory(Object target) {
this.target = target;
}
//給目標對象創建一個代理對象
public Object getProxyInstance() {
// 1.工具類
Enhancer en = new Enhancer();
// 2. 設置父類
en.setSuperclass(target.getClass());
// 3. 設置回調函數
en.setCallback(this);
// 4. 創建子類(代理對象)
return en.create();
}
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("開始事務3");
Object returevalue = method.invoke(target,objects);
System.out.println("提交事務3");
return returevalue;
}
}
~~~
### 測試類
~~~
package com.yc.model;
import com.yc.model.Proxy.UserDaoProxy;
import com.yc.model.ProxyFactory.CglibProxyFacory;
import com.yc.model.Service.IUserDao;
import com.yc.model.Service.Impl.UserDao;
public class Test {
public static void main (String[] args){
//目標對象
IUserDao target =new UserDao();
//代理對象
IUserDao proxy = (IUserDao)new CglibProxyFacory(target).getProxyInstance();
//執行的是代理的方法
proxy.save();
}
}
~~~
### 結果
