Java 动态代理

更多 Java 高级知识方面的文章,请参见文集《Java 高级知识》


代理模式

  • 委托类:具体提供服务
  • 代理类:通过调用委托类对象的方法来提供服务。代理类的作用包括:
    • 调用具体方法前进行预处理,包括:
      • 数据的验证及过滤
      • 权限的验证
      • 开启事务
      • 查询缓存
      • 建立数据库连接
      • 记录日志等等。
    • 调用具体方法后进行后处理,包括:
      • 提交事务
      • 更新缓存
      • 关闭数据库连接
      • 记录日志等等。

代理模式分为两种形式:

  • 静态代理:程序运行前,代理类已经存在于 class 字节码文件中,程序运行中只是创建代理类的对象。
  • 动态代理:程序运行中动态创建出代理类及其对象(原理:通过反射来动态产生字节码)

静态代理

示例如下:

public class Proxy_Test {
    public static void main(String[] args) throws Exception {
        AddProxy proxy = new AddProxy(new AddImpl());
        proxy.add(1, 2);
    }
}

// 共同的接口
interface Add {
    int add(int a, int b);
}

// 委托类
class AddImpl implements Add {
    public int add(int a, int b) {
        return a + b;
    }
}

// 代理类
class AddProxy implements Add {
    // 代理类关联一个委托类对象
    private AddImpl impl;

    // 通过构造方法传入委托类的对象
    public AddProxy(AddImpl impl) {
        this.impl = impl;
    }

    public int add(int a, int b) {
        // 预处理,记录日志
        System.out.println("Add start");

        // 通过调用委托类对象的方法来提供服务
        return impl.add(a, b);
    }
}

可以看出:

  • 委托类和代理类需要实现同一个接口
  • 委托类具体提供服务
  • 代理类通过调用委托类对象的方法来提供服务,在调用前进行预处理操作,例如记录日志
  • 代理类关联一个委托类对象,并通过构造方法传入委托类的对象

静态代理的问题

代理类与委托类一一对应。
如果有多个委托类,例如在上述的例子中存在 AddImpl MinusImpl MultiplyImpl DivideImpl,则需要对应地构造多个代理类,产生重复的代码,因为每个代理类的功能其实都一样,例如记录日志。

动态代理

一个代理类完成全部的代理功能,代理一个或多个委托类。

Proxy 类

所在包:import java.lang.reflect.Proxy;
通过 Proxy 类的 newProxyInstance 方法可以为一个或多个接口动态地创建出代理类及其对象。

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

该方法返回一个代理类对象,该对象实现了指定的接口 Class<?>[] interfaces

newProxyInstance 方法中,会调用如下代码:即在运行时动态生成代理类的字节码,并将字节码转换为代理类的对象。

/*
 * Generate the specified proxy class.
 */
byte[] proxyClassFile = ProxyGenerator.generateProxyClass(
    proxyName, interfaces, accessFlags);

InvocationHandler 接口

所在包:import java.lang.reflect.InvocationHandler;
实现如下的 invoke 方法:

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

Processes a method invocation on a proxy instance and returns the result. This method will be invoked on an invocation handler when a method is invoked on a proxy instance that it is associated with.
当调用代理类对象的方法时,会自动调用该 invoke 方法。
在该 invoke 方法中进行预处理,调用委托类方法 method,后处理,返回结果。

通过动态代理类实现如上的静态代理功能,完整的代码如下:

public class Proxy_Test {
    public static void main(String[] args) throws Exception {
        // 动态创建 AddImpl 的代理类对象
        Add addProxy = (Add) generateProxy(new AddImpl());
        addProxy.add(1, 2);

        // 动态创建 MinusImpl 的代理类对象
        Minus minusProxy = (Minus) generateProxy(new MinusImpl());
        minusProxy.minus(1, 2);
    }

    public static Object generateProxy(Object obj) {
        // 通过 Proxy 类的 newProxyInstance 方法可以为一个或多个接口动态地创建出代理类及其对象
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(),
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        // 预处理,记录日志
                        System.out.println(method.getName() + " start");

                        // 实际调用委托类的具体方法
                        return method.invoke(obj, args);
                    }
                });
    }
}

// 共同的接口
interface Add {
    int add(int a, int b);
}

interface Minus {
    int minus(int a, int b);
}

interface Multiply {
    int multiply(int a, int b);
}

interface Divide {
    int divide(int a, int b);
}

// 委托类
class AddImpl implements Add {
    public int add(int a, int b) {
        return a + b;
    }
}

class MinusImpl implements Minus {
    public int minus(int a, int b) {
        return a - b;
    }
}

class MultiplyImpl implements Multiply {
    public int multiply(int a, int b) {
        return a * b;
    }
}

class DivideImpl implements Divide {
    public int divide(int a, int b) {
        return a / b;
    }
}

动态代理的原理

调用代理类的方法时 addProxy.add(1, 2);,会自动触发 invoke 方法,invoke 方法中调用委托类的方法 method.invoke(obj, args);
实际上,invoke 方法不是显示调用的,它是一个回调函数。

动态代理的问题

委托类需要实现某一个接口,例如 AddImpl 需要实现 AddMinusImpl 需要实现 Minus

CGLib 可以弥补这个缺陷,不需要实现特定的接口。
CGLib 通过继承来实现,动态产生的代理类实际上为委托类的子类,代理类通过 Method Override 实现了代码增强。
例如:

Enhancer e = new Enhancer();
e.setSuperclass(AddImpl.class);
e.setCallback(new MethodInterceptor() {
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        // 预处理,记录日志
        System.out.println(method.getName() + " start");

        return method.invoke(o, objects);
    }
});

// 动态产生的代理类实际上为委托类的子类
AddImpl proxy = (AddImpl)e.create();
proxy.add(1, 2);

关于 CGLib 和 JDK Proxy 的比较,可以参见 Stack Overflow

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

推荐阅读更多精彩内容