Java动态代理

在工作之余看一些优秀源码的时候发现很多地方使用了动态代理,所以抽了一些时间对java的动态代理深入熟悉一下,这篇文章记录我学习java动态代理的一些足迹,本篇文章的主要内容如下:
jdk动态代理
cglib动态代理

jdk动态代理

首先我们通过一个例子来回顾一下如何实现一个jdk动态代理
1.定义一个业务接口,并且实现它

public interface IFunction {
    int function(int a, int b);
    void print();
}

public class FunctionImpl implements IFunction {
    @Override
    public int function(int a, int b) {
        return a + b;
    }
    @Override
    public void print() {
        System.out.println("FunctionImpl print!");
    }
}

2.通过实现 java.lang.reflect.InvocationHandler 来完成一个代理类, 并且使用java.lang.reflect.Proxy 的newProxyInstance方法来获取代理对象

public class FunctionProxy implements InvocationHandler{

    private IFunction target;

    public FunctionProxy(IFunction function) {
        this.target = function;
    }

    //获取代理对象
    public Object getProxy() {
        return Proxy.newProxyInstance(Thread.currentThread()
                        .getContextClassLoader(), target.getClass().getInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        before();
        Object obj = method.invoke(target,args); //注意点1
        after();
        return obj;
    }

    private void before(){
        System.out.println("before!");
    }

    private void after(){
        System.out.println("after!");
    }
}

3.使用jdk动态代理

public class JdkProxyEntrance {

    public static void main(String[] args) {
        IFunction function = new FunctionImpl();
        FunctionProxy proxy1 = new FunctionProxy(function);
        IFunction obj = (IFunction)getProxy(function, proxy1);
        int result = obj.function(3,2);
        System.out.println(result);
    }
}

通过上面的例子我们已经能够跑起来一个jdk动态代理,接下来就要探寻一下其中的原理,那么我们的入口毫无疑问,就是Proxy 的 newProxyInstance 方法,我们很好奇它是怎么动态生成一个代理对象的。

public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h)

进入上面的那个方法,其核心实现如下:

/*
* Look up or generate the designated proxy class. 
*/
//①
Class<?> cl = getProxyClass0(loader, interfaces);

//②
final Constructor<?> cons = cl.getConstructor(constructorParams);
return cons.newInstance(new Object[] {h} );

代码简析:
①先获取代理类的 Class 对象
②通过反射创建一个此Class的实例

对于反射获取实例这个不用说,点开我们不熟悉的getProxyClass0方法,其核心代码如下:

//proxyName:系统设定的代理类名字,格式 "com.sun.proxy.$Proxy1"
//interfaces:被代理类实现的所有接口
byte[] proxyClassFile = ProxyGenerator.generateProxyClass( proxyName, interfaces);
Class<?> proxyClass = defineClass0(loader, proxyName,proxyClassFile, 0, proxyClassFile.length);

代码简析:
①根据给定参数,使用字节码技术生成动态代理类的字节码(这其中还是挺复杂的)
②调用本地方法 defineClass0 生成Class对象

到了这儿读者都明白了jdk动态代理的大概流程是通过字节码技术动态生成了一个动态代理类的Class对象,然后利用反射创建一个代理类的实例给用户使用。这时候我们就想要是能看看这个代理类的字节码,也就是这个代理类到底长什么样就更好了。

运行如下代码来获得动态代理类的 .class文件

public static void generaterClassFile(Class[] interfaces){
    String proxyName = "com.sun.proxy.$Proxy0";
    byte[] proxyClassFile = ProxyGenerator.generateProxyClass(
            proxyName, interfaces);
    try {
        FileOutputStream var1 = new FileOutputStream(proxyName + ".class");
        var1.write(proxyClassFile);
        var1.close();
    } catch (IOException var2) {
        throw new InternalError("I/O exception saving generated file: " + var2);
    }
}

ps:也可以通过设置如下的代码来自动保留生成的字节码,只是我在试验的报了FileNotFoundException错误 ,后面我会找找原因,如果有热心的读者给我指点出来我也很高兴。

System.setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");

细心的你会发现上面 generaterClassFile 方法中生成字节码的方式其实就是和jdk动态代理生成字节码的方式一样,使用 ProxyGenerator.generateProxyClass 方法,生成的 .class文件使用java 反编译工具jd 反编译后的代码如下:

package com.sun.proxy;

import com.zcd.learn.designpattern.proxy.IFunction;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;

public final class $Proxy0 extends Proxy
  implements IFunction
{
  private static Method m1;
  private static Method m4;
  private static Method m0;
  private static Method m3;
  private static Method m2;

  public $Proxy0(InvocationHandler paramInvocationHandler)
    throws 
  {
    super(paramInvocationHandler);
  }

  public final boolean equals(Object paramObject)
    throws 
  {
    try
    {
      return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final int function(int paramInt1, int paramInt2)
    throws 
  {
    try
    {
      return ((Integer)this.h.invoke(this, m4, new Object[] { Integer.valueOf(paramInt1), Integer.valueOf(paramInt2) })).intValue();
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final int hashCode()
    throws 
  {
    try
    {
      return ((Integer)this.h.invoke(this, m0, null)).intValue();
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final void print()
    throws 
  {
    try
    {
      this.h.invoke(this, m3, null);
      return;
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final String toString()
    throws 
  {
    try
    {
      return (String)this.h.invoke(this, m2, null);
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  static
  {
    try
    {
      m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") });
      m4 = Class.forName("com.zcd.learn.designpattern.proxy.IFunction").getMethod("function", new Class[] { Integer.TYPE, Integer.TYPE });
      m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
      m3 = Class.forName("com.zcd.learn.designpattern.proxy.IFunction").getMethod("print", new Class[0]);
      m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
      return;
    }
    catch (NoSuchMethodException localNoSuchMethodException)
    {
      throw new NoSuchMethodError(localNoSuchMethodException.getMessage());
    }
    catch (ClassNotFoundException localClassNotFoundException)
    {
      throw new NoClassDefFoundError(localClassNotFoundException.getMessage());
    }
  }
}

我们会发现这个动态代理类会持有我们的代理类实例(h),无论是调用function 方法还是 print 方法都是调用我们的代理类FunctionProxy 的invoke 方法(this.h.invoke(this, m3, null)),并且传入本方法的Method对象,而在FunctionProxy 的 invoke方法则是采用反射来调用委托类的具体方法。

最后记录一个我遇到的一个问题:
把FunctionProxy 中 invoke 方法中我标注的注意点1 换成如下代码,会出现死循环:

Object obj = method.invoke(proxy,args);

原因就是proxy就是代理类实例本身,这样写就是重复调用自己会出现死循环。也是在反编译后我才彻底明白为什么会有这样的错误,也让我对jdk动态代理理解更深刻。

Cglib动态代理

简介:
在我们上一节中使用jdk动态代理,必须实现一个接口,而在实际的生产环境中并不是所有的类都是实现了某某接口的,这时候我们要实现这种类的代理就可以使用cglib动态代理。它是通过动态生成一个子类继承于被代理类,从而达到代理目的。

老规矩通过一个例子来看看如何实现一个cglib动态代理。
cglib动态需要引入外部jar包,我通过maven引入方式如下:

<!-- https://mvnrepository.com/artifact/cglib/cglib -->
<dependency>
  <groupId>cglib</groupId>
  <artifactId>cglib</artifactId>
  <version>3.1</version>
</dependency>

1.定义被代理类

public class CglibFunction {
    public void funcA(){
        System.out.println("CglibFunction funcA");
    }

    public int funcB(int a, int b){
        System.out.println("CglibFunction funcB");
        return a + b;
    }
}

2.通过实现 net.sf.cglib.proxy.MethodInterceptor 接口来实现cglib代理类

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

/**
 * Created by zhangyida on 2017/9/15.
 */
public class CglibProxy implements MethodInterceptor {

    public Object intercept(Object obj, Method method, Object[] args,
                            MethodProxy proxy) throws Throwable {
        System.out.println("cglib before!");
        //通过代理类调用父类中的方法
        Object result = proxy.invokeSuper(obj, args);
        System.out.println("cglib after!");
        return result;
    }
}

3.通过net.sf.cglib.proxy.Enhancer 来动态获取代理实例

CglibProxy proxy = new CglibProxy();
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(CglibFunction.class);
enhancer.setCallback(proxy);
CglibFunction proxyImp = (CglibFunction)enhancer.create();
proxyImp.funcA();

运行结果如下:

cglib before!
CglibFunction funcA
cglib after!

Enhancer是一个字节码增强器,下面是cglib源代码中对Enhancer作用的描述:
Generates dynamic subclasses to enable method interception.
其生成动态代理子类的核心代码如下:

byte[] b = strategy.generate(this);
String className = ClassNameReader.getClassName(new ClassReader(b));
getClassNameCache(loader).add(className);
Class gen = ReflectUtils.defineClass(className, b, loader);

故事总是惊人的相似,cglib动态获取代理实例的步骤如下:
①生成代理类的字节码二进制数组
②根据字节码获取Class对象
③利用反射机制创建代理类的实例。

接下来可以通过如下的方法来生成动态代理类的.class文件,因为在net.sf.cglib.core.DebuggingClassWriter 中给我们预留了这个功能,有兴趣的读者可以去看看,这个类中的toByteArray 方法。

1.在代码中加入

System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "E:\\Bode");

2.或者在vm 启动参数参数中添加如下参数:

-Dcglib.debugLocation="E:\\Bode"

这两种方法均能达到在E盘的Bode目录下生成代理类的.class,用反编译工具(有感兴趣的读者推荐使用procyon-decompiler,因为我用jd反编译有很多缺失)打开这些 .class文件,其中较重要的内容展示如下(由于内容比较多,删除了一些无关紧要的代码):

// 
// Decompiled by Procyon v0.5.29
// 

package com.zcd.learn.designpattern.proxy.cglibproxy;

import net.sf.cglib.core.Signature;
import net.sf.cglib.core.ReflectUtils;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.Callback;
import net.sf.cglib.proxy.Factory;

public class CglibFunction$$EnhancerByCGLIB$$97816eeb extends CglibFunction implements Factory
{
    private MethodInterceptor CGLIB$CALLBACK_0;
    private static final Method CGLIB$funcA$0$Method;
    private static final MethodProxy CGLIB$funcA$0$Proxy;
    private static final Method CGLIB$funcB$1$Method;
    private static final MethodProxy CGLIB$funcB$1$Proxy;
    
    static void CGLIB$STATICHOOK1() {
        final Method[] methods2 = ReflectUtils.findMethods(new String[] { "funcA", "()V", "funcB", "(II)I" }, (forName3 = Class.forName("com.zcd.learn.designpattern.proxy.cglibproxy.CglibFunction")).getDeclaredMethods());
        CGLIB$funcA$0$Method = methods2[0];
        CGLIB$funcA$0$Proxy = MethodProxy.create((Class)forName3, (Class)forName, "()V", "funcA", "CGLIB$funcA$0");
        CGLIB$funcB$1$Method = methods2[1];
        CGLIB$funcB$1$Proxy = MethodProxy.create((Class)forName3, (Class)forName, "(II)I", "funcB", "CGLIB$funcB$1");
    }
    
    final void CGLIB$funcA$0() {
        super.funcA();
    }
    
    public final void funcA() {
        MethodInterceptor cglib$CALLBACK_;
        MethodInterceptor cglib$CALLBACK_0;
        if ((cglib$CALLBACK_0 = (cglib$CALLBACK_ = this.CGLIB$CALLBACK_0)) == null) {
            CGLIB$BIND_CALLBACKS(this);
            cglib$CALLBACK_ = (cglib$CALLBACK_0 = this.CGLIB$CALLBACK_0);
        }
        if (cglib$CALLBACK_0 != null) {
            cglib$CALLBACK_.intercept((Object)this, CglibFunction$$EnhancerByCGLIB$$97816eeb.CGLIB$funcA$0$Method, CglibFunction$$EnhancerByCGLIB$$97816eeb.CGLIB$emptyArgs, CglibFunction$$EnhancerByCGLIB$$97816eeb.CGLIB$funcA$0$Proxy);
            return;
        }
        super.funcA();
    }
    
    final int CGLIB$funcB$1(final int n, final int n2) {
        return super.funcB(n, n2);
    }
    
    public final int funcB(final int n, final int n2) {
        MethodInterceptor cglib$CALLBACK_;
        MethodInterceptor cglib$CALLBACK_0;
        if ((cglib$CALLBACK_0 = (cglib$CALLBACK_ = this.CGLIB$CALLBACK_0)) == null) {
            CGLIB$BIND_CALLBACKS(this);
            cglib$CALLBACK_ = (cglib$CALLBACK_0 = this.CGLIB$CALLBACK_0);
        }
        if (cglib$CALLBACK_0 != null) {
            final Object intercept = cglib$CALLBACK_.intercept((Object)this, CglibFunction$$EnhancerByCGLIB$$97816eeb.CGLIB$funcB$1$Method, new Object[] { new Integer(n), new Integer(n2) }, CglibFunction$$EnhancerByCGLIB$$97816eeb.CGLIB$funcB$1$Proxy);
            return (intercept == null) ? 0 : ((Number)intercept).intValue();
        }
        return super.funcB(n, n2);
    }
    
    public static MethodProxy CGLIB$findMethodProxy(final Signature signature) {
        final String string = signature.toString();
        switch (string.hashCode()) {
            case -1555444918: {
                if (string.equals("funcB(II)I")) {
                    return CglibFunction$$EnhancerByCGLIB$$97816eeb.CGLIB$funcB$1$Proxy;
                }
                break;
            }
            case 1379354712: {
                if (string.equals("funcA()V")) {
                    return CglibFunction$$EnhancerByCGLIB$$97816eeb.CGLIB$funcA$0$Proxy;
                }
                break;
            }
        }
        return null;
    }

    static {
        CGLIB$STATICHOOK1();
    }
}

以上就是cglib动态代理生成的字节码最需要我们关注的部分,我们得出结论:
1.此动态生成的类继承于需要被代理的类(委托类)

public class CglibFunction$$EnhancerByCGLIB$$97816eeb extends CglibFunction implements Factory

2.此类持有我们编写的代理类的实例(经过调试确认),因为其有一个私有属性,这个属性被引用的地方很多

private MethodInterceptor CGLIB$CALLBACK_0;

3.此代理类为每个方法(无论是Object还是CglibFunction的方法),都对应着两个方法,对于funcA 方法对应 CGLIB$funcA$0 和 funcA。这个同名的方法funcA会直接调用我们写的代理类的intercept 方法,CGLIB$funcA$0 方法则是直接调用委托类的目标方法 funcA。
4.此代理类为每个方法都构建了一个MethodProxy 对象

private static final MethodProxy CGLIB$funcB$1$Proxy;
CGLIB$funcB$1$Proxy = MethodProxy.create((Class)forName3, (Class)forName, "(II)I", "funcB", "CGLIB$funcB$1");

5.为什么要提这个MethodProxy呢,因为我们自己实现的代理类的intercept方法中最重要的一步就是

Object result = proxy.invokeSuper(obj, args);

而这个proxy就是一个MethodProxy 实例,点开这个invokeSuper方法

public Object invokeSuper(Object obj, Object[] args) throws Throwable {
    try {
        init();
        FastClassInfo fci = fastClassInfo;
        return fci.f2.invoke(fci.i2, obj, args);
    } catch (InvocationTargetException e) {
        throw e.getTargetException();
    }
}

接下来有两个重要的类叫FastClassInfo, 和 FastClass

private static class FastClassInfo
{
    FastClass f1;   //指向委托类的Class
    FastClass f2;  //指向动态代理类的Class
    int i1;   //funcA在代理对象中的索引位置
    int i2;  //CGLIB$funcA$0 在代理对象中的索引位置
}

abstract public class FastClass
{
    private Class type;
    //省略很多
}

那么我们就可以知道invokeSuper 方法中

fci.f2.invoke(fci.i2, obj, args);

就是调用的父类的同名方法,在MethodProxy 类中还有invoke 方法,

return fci.f1.invoke(fci.i1, obj, args);

读者可以试试调用这个方法,也会出现在jdk动态代理最后我提的死循环问题。

总结:
无论是jdk动态代理还是 cglib 动态代理采用字节码技术生成了一个新的代理类,这个动态代理类持有我们编写的代理类的实例,我们调用目标方法其实都是调用InvocationHandler 的 invoke 方法,或者MethodInterceptor 的 intercept 方法,然后通过一定的技术调用委托类的方法,一个采用反射,一个采用类似数组下标的方式来定位方法,直接进行类方法的执行。

后记:
此次是本人第一次写技术博客,个中出现的不正确或者表达不恰当或者不明白之处望热心读者可以给我指正。同时我也认为博客能记录自己的学习足迹,也能与广大读者共同交流技术‘思想’,在这里自勉自己能一直坚持下去。

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

推荐阅读更多精彩内容