动态代理技术是整个java技术系统中非常重要的一环,它是我们能够深入学习java框架的基础,是深入了解Spring等框架时要掌握的基础知识之一。
Java中自带的动态代理的类必须要实现一个接口,也就是说只能对该类所实现接口中定义的方法进行代理,这在实际编程中具有一定的局限性,而且使用反射的效率也并不是很高。于是CGLIB就诞生了。
使用CGLib实现动态代理,完全不受代理类必须实现接口的限制,而且CGLib底层采用ASM字节码生成框架,使用字节码技术生成代理类,理论上比使用Java反射效率要高。
1、 CGLib动态代理的概念
举例:歌星或者明星都有一个自己的经纪人,这个经纪人就是他们的代理人,当我们需要找明星表演时,不能直接找到该明星,只能是找明星的代理人。比如王宝强在现实生活中非常有名,会唱歌,会跳舞,会拍戏。王宝强在没有出名之前,在他们村种地的时候,我们可以直接找他唱歌,跳舞,拍戏,王宝强出名之后,他找了一个经纪人,这个经纪人就是王宝强的代理人(代理),当我们需要找王宝强表演时,不能直接找到王宝强了(王宝强说,你找我的代理人吧!),因此王宝强这个代理人存在的价值就是拦截我们对王宝强的直接访问!
原来的代理人要求和王宝强一样会唱歌跳舞,但是这样的代理人不多呀,实在找不到这样的代理人,那么就降低要求把,你要找王宝强,我就把他找出来给你好了。
王宝强(真实业务对象)会唱歌,会跳舞,会拍戏,我们现在不能直接找他唱歌,跳舞,拍戏了,只能找他的代理人(代理对象)唱歌,跳舞,拍戏。
2、 一些前提条件
需要额外导入cglib-2.2.0.jar
3、 Java中利用CGLib实现目标对象类(具体实施对象)
package com.CGLib;
/**
* 王宝强是一个演员,他会唱歌和跳舞.
* 这里没有实现接口
* @author 范芳铭
*/
public class WangBaoQiang{
private String name ;
public WangBaoQiang(){
this.name = "王宝强";
}
public String sing(String name){
System.out.println(this.getName() + "开始唱"+name+"歌!!");
return "歌唱完了,谢谢大家!";
}
public String dance(String name){
System.out.println(this.getName() + "开始跳"+name+"舞!!");
return "舞跳完了,谢谢大家!";
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
4、 JAVA中利用CGLib生成代理对象的代理类
package com.CGLib;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
/**
* 演员的经纪人,他不会唱歌和跳舞,但是经纪人能找到会唱歌跳舞的人
* @author 范芳铭
*/
public class ActorJingJiRen implements MethodInterceptor{
public Object intercept(Object obj, Method method, Object[] arg, MethodProxy proxy) throws Throwable {
System.out.println("CGLIB Before:"+method);
System.out.println("我是他的经纪人,有事情先找我!");
Object object=proxy.invokeSuper(obj, arg);
System.out.println("CGLIB After: 结束。");
return object;
}
public WangBaoQiang getProxy(){
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(WangBaoQiang.class);
enhancer.setCallback(new ActorJingJiRen());
WangBaoQiang object=(WangBaoQiang)enhancer.create();
return object;
}
}
5、 调用动态代理的测试类
package com.CGLib;
/**
* 测试类,找到经纪人提出唱歌跳舞的要求就行
*
* @author 范芳铭
*/
public class ProxyTest {
public static void main(String[] args) {
// 首先找到经纪人
ActorJingJiRen proxy = new ActorJingJiRen();
// 通过经纪人获得相关演员(代理对象)
WangBaoQiang p = proxy.getProxy();
// 让演员唱歌
String retValue = p.sing("天下无贼");
System.out.println(retValue);
// 让演员跳舞
String value = p.dance("凤凰传奇");
System.out.println(value);
}
}
6、 运行结果
CGLIB Before:public java.lang.String com.CGLib.WangBaoQiang.sing(java.lang.String)
我是他的经纪人,有事情先找我!
CGLIB Before:public java.lang.String com.CGLib.WangBaoQiang.getName()
我是他的经纪人,有事情先找我!
CGLIB After: 结束。
王宝强开始唱天下无贼歌!!
CGLIB After: 结束。
歌唱完了,谢谢大家!
CGLIB Before:public java.lang.String com.CGLib.WangBaoQiang.dance(java.lang.String)
我是他的经纪人,有事情先找我!
CGLIB Before:public java.lang.String com.CGLib.WangBaoQiang.getName()
我是他的经纪人,有事情先找我!
CGLIB After: 结束。
王宝强开始跳凤凰传奇舞!!
CGLIB After: 结束。
舞跳完了,谢谢大家!
补充说明:在执行一个动作之前,有2个CGLIB Before被执行,是因为还有一个内部动作被执行(getName()),因为所有的动作都会被拦截,因此这里有2个。