【黑馬程序員濟南中心】代理模式-Cglib代理
今天和你們聊一條java中代理模式的第三種設計模式-Cglib模式。前兩天咱們聊了靜態代理和動態代理,可是你們有沒有發現這兩種代理模式都是要求目標對象是實現一個接口的目標對象,可是有時候目標對象只是一個單獨的對象,並無實現任何的接口。而這時候咱們就可使用Cglib模式,它可使用以目標對象子類的方式類實現代理。java
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,那麼就不會被攔截,即不會執行目標對象額外的業務方法.ide
下邊給你們舉個使用Cglib代理的例子:函數
目標對象類:UserDao.java:工具
/**性能
* 目標對象,沒有實現任何接口測試
*/
public class UserDao {
public void save() {
System.out.println("----已經保存數據!----");
}
}
Cglib代理工廠:ProxyFactory.java:
/**
* Cglib子類代理工廠
* 對UserDao在內存中動態構建一個子類對象
*/
public class ProxyFactory implements MethodInterceptor{
//維護目標對象
private Object target;
public ProxyFactory(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();
}
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("開始事務...");
//執行目標對象的方法
Object returnValue = method.invoke(target, args);
System.out.println("提交事務...");
return returnValue;
}
}
測試類:
/**
* 測試類
*/
public class App {
public void test(){
//目標對象
UserDao target = new UserDao();
//代理對象
UserDao proxy = (UserDao)new ProxyFactory(target).getProxyInstance();
//執行代理對象的方法
proxy.save();
}
}
在Spring的AOP編程中,若是加入容器的目標對象有實現接口,用JDK代理,若是目標對象沒有實現接口,用Cglib代理。
通常Cglib代理在Spring框架中應用,因此小夥伴們能夠把它當作瞭解內容看一看。