Java Proxy和CGLIB动态代理原理

如果觉得再简述上阅读代码太困难可以点这里:Spring中动态代理详解

动态代理在Java中有着广泛的应用,比如Spring AOP,Hibernate数据查询、测试框架的后端mock、RPC,Java注解对象获取等。静态代理的代理关系在编译时就确定了,而动态代理的代理关系是在编译期确定的。静态代理实现简单,适合于代理类较少且确定的情况,而动态代理则给我们提供了更大的灵活性。今天我们来探讨Java中两种常见的动态代理方式:JDK原生动态代理和CGLIB动态代理。

    JDK原生动态代理

先从 直观的示例说起,假设我们有一个接口(interface Hello)和一个简单的实现类(HelloImp)

// 接口

interface Hello{

    String sayHello(String str);

}

// 实现

class HelloImp implements Hello{

    @Override

    public String sayHello(String str) {

      return "HelloImp: " + str;

    }

}


这是java中再常见不过的场景了,使用接口制定协议,然后用不同的实现来实现具体行为。假设你已经拿到上述类库,如果我们通过日志记录对sayHello()的调用,使用静态代理可以这样做:

// 静态代理方式

class StaticProxiedHello implements Hello{

...

private Hello hello = new HelloImp();

@Override

public String sayHello(String str) {

logger.info("You said: " + str);

return hello.sayHello(str);

}

}

上述的静态代理类StaticProxiedHello做为HelloImp的代理,实现了相同的Hello接口,使用java动态代理可以这样做:

  1.首先实现一个InvocationHandler,方法调用会被转发到该类的invoke()方法。

  2.然后在需要使用Hello的时候,通过JDK动态代理获取Hello的代理对象。

// Java Proxy

// 1. 首先实现一个InvocationHandler,方法调用会被转发到该类的invoke()方法。

class LogInvocationHandler implements InvocationHandler{

...

private Hello hello;

public LogInvocationHandler(Hello hello) {

this.hello = hello;

}

@Override

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

if("sayHello".equals(method.getName())) {

logger.info("You said: " + Arrays.toString(args));

}

return method.invoke(hello, args);

}

}

// 2. 然后在需要使用Hello的时候,通过JDK动态代理获取Hello的代理对象。

Hello hello = (Hello)Proxy.newProxyInstance( getClass().getClassLoader(), // 1. 类加载器

new Class<?>[] {Hello.class}, // 2. 代理需要实现的接口,可以有多个

new LogInvocationHandler(new HelloImp()));// 3. 方法调用的实际处理者

System.out.println(hello.sayHello("I love you!"));

运行上述代码输出结果:

日志信息: You said: [I love you!]

HelloImp: I love you!

上述代码的关键是Proxy.newProxyInstance(ClassLoder loder,Class<?> interfance,InvocationHandler handler)方法,该方法会根据指定的参数动态创建代理对象,三个参数的意义如下:

  1.loder:指定代理对象的类加载器;

  2.interface:代理对象需要实现的接口,可以同时指定多个接口;

  3.handler:方法调用的实际处理者,代理对象的方法调用都会转发到这里。

newProxyInstance()会返回一个实现了指定接口的代理对象,对该对象的所有方法调用都会转发给InvocationHandler.invoke()方法,理解上述代码需要对java的反射机制有一定的了解动态代理的神奇之处在于:

  1.代理对象是在程序运行时产生的,而不是编译期;

  2.对代理对象的所有接口方法调用都会转发到InvcationHandler.invoke()方法,在invoke()方法里我们可以加入任何逻辑,比如修改方法参数,加入日志功能,安全检查功能等;之后我们通过某种方式执行真正的方法体,实例中通过反射调用了Hello对象的相应方法。还可以通过RPC调用远程方法。

注意1:对于从Object中继承的方法,JDK Proxy会把hashCode()、equals()、toString()这三个非接口方法转发给InvocationHandler,其余的Object方法则不会转发。详见JDK Proxy官方文档。

如果对JDK代理后的对象类型进行深挖,可以看到如下信息:

# Hello代理对象的类型信息

class=class jdkproxy.$Proxy0

superClass=class java.lang.reflect.Proxy

interfaces: interface jdkproxy.Hello

invocationHandler=jdkproxy.LogInvocationHandler@a09ee92

代理对象的类型是jdkproxy.$Proxy0,这个是动态生成的类型,类名是形如$ProxyN的形式;父类是java.lang.reflect.Proxy,所有的JDK动态代理都会继承这个类;同时实现了Hello接口也就是我们接口列表中指定的那些接口。

jdkproxy.$Proxy0具体实现:

// JDK代理类具体实现

public final class $Proxy0 extends Proxy implements Hello {

...

public $Proxy0(InvocationHandler invocationhandler) {

super(invocationhandler);

}

...

@Override

public final String sayHello(String str){

...

return super.h.invoke(this, m3, new Object[] {str});// 将方法调用转发给invocationhandler

...

}

...

}


这些逻辑没什么复杂之处,但是他们是在运行时动态产生的,无需我们手动编写。更多详情,可参考BrightLoong的Java静态代理&动态代理笔记

Java动态代理为我们提供了非常灵活的代理机制,但Java动态代理是基于接口的,如果对象没有实现接口我们该如何代理呢?CGLIB登场。




CGLIB(Code Generation Library)是一个基于ASM的字节码生成库,它允许我们在运行时对字节码进行修改和动态生成。CGLIB通过继承方式实现代理。

来看示例,假设我们有一个没有实现任何接口的类HelloConcrete:

public class HelloConcrete {

public String sayHello(String str) {

return "HelloConcrete: " + str;

}

}

因为没有实现接口该类无法使用JDK代理,通过CGBL代理实现如下:

  1.首先实现一个MethodInterceptor,方法调用会被转发到该类的intercept()方法。

  2.然后再需要使用HelloConcrete的时候通过CGLB动态代理获得代理对象。

// CGLIB动态代理

// 1. 首先实现一个MethodInterceptor,方法调用会被转发到该类的intercept()方法。

class MyMethodInterceptor implements MethodInterceptor{

...

@Override

public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {

logger.info("You said: " + Arrays.toString(args));

return proxy.invokeSuper(obj, args);

}

}

// 2. 然后在需要使用HelloConcrete的时候,通过CGLIB动态代理获取代理对象。

Enhancer enhancer = new Enhancer();

enhancer.setSuperclass(HelloConcrete.class);

enhancer.setCallback(new MyMethodInterceptor());

HelloConcrete hello = (HelloConcrete)enhancer.create();

System.out.println(hello.sayHello("I love you!"));

运行上述代码输出结果:

日志信息: You said: [I love you!]

HelloConcrete: I love you!

上述代码中,我们通过CGLIB的Enhancer 来指定要代理的目标对象、实际处理代理逻辑的对象,最终通过调用create() 方法得到代理对象,对这个对象所有非final方法的调用都会转发给MethodInterceptor.intercept() 方法 ,在intercept()方法里我们可以加入任何逻辑,比如修改方法参数,加入日志功能、安全检查功能等;通过调用MethodProxy.invokeSuper()方法,我们将调用转发给原始对象,具体到本例,就是HelloConcrete的具体方法。CGLIG中MethodInterceptor的作用跟JDK代理中的InvocationHandler很类似,都是方法调用的中转站。

注意:对于从Object中继承的方法,CGLIB代理也会进行代理,如hashCode()、equals()、toString()等,但是getClass()、wait()等方法不会,因为它是final方法,CGLIB无法代理。

如果对CGLIB代理之后的对象类型进行深挖,可以看到如下信息:

# HelloConcrete代理对象的类型信息

class=class cglib.HelloConcrete

EnhancerByCGLIB

EnhancerByCGLIB

e3734e52

superClass=class lh.HelloConcrete

interfaces:

interface net.sf.cglib.proxy.Factory

invocationHandler=not java proxy class

我们看到使用CGLIB代理之后的对象类型是cglib.HelloConcrete

EnhancerByCGLIB

EnhancerByCGLIB

e3734e52,这是CGLIB动态生成的类型;父类是HelloConcrete,印证了CGLIB是通过继承实现代理;同时实现了net.sf.cglib.proxy.Factory接口,这个接口是CGLIB自己加入的,包含一些工具方法。

注意,既然是继承就不得不考虑final的问题。我们知道final类型不能有子类,所以CGLIB不能代理final类型,遇到这种情况会抛出类似如下异常:

java.lang.IllegalArgumentException: Cannot subclass final class cglib.HelloConcrete

同样的,final方法是不能重载的,所以也不能通过CGLIB代理,遇到这种情况不会抛异常,而是会跳过final方法只代理其他方法。

如果你还对代理类cglib.HelloConcrete

EnhancerByCGLIB

EnhancerByCGLIB

e3734e52具体实现感兴趣,它大致长这个样子:

// CGLIB代理类具体实现

public class HelloConcrete$$EnhancerByCGLIB$$e3734e52 extends HelloConcrete implements Factory {

...

private MethodInterceptor CGLIB$CALLBACK_0; // ~~

...

public final String sayHello(String paramString) {

...

MethodInterceptor tmp17_14 = CGLIB$CALLBACK_0;

if (tmp17_14 != null) { // 将请求转发给MethodInterceptor.intercept()方法。

return (String)tmp17_14.intercept(this, CGLIB$sayHello$0$Method, new Object[] { paramString }, CGLIB$sayHello$0$Proxy);

}

return super.sayHello(paramString);

}

...

}

上述代码我们看到,当调用代理对象的sayHello()方法时,首先会尝试转发给MethodInterceptor.intercept()方法,如果没有MethodInterceptor就执行父类的sayHello()。这些逻辑没什么复杂之处,但是他们是在运行时动态产生的,无需我们手动编写。如何获取CGLIB代理类字节码可参考Access the generated byte[] array directly。

更多关于CGLIB的介绍可以参考Rafael Winterhalter的cglib: The missing manual,一篇很深入的文章。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352