Java 静态代理和动态代理的使用及原理解析

代理模式是软件开发中常见的设计模式,它的目的是让调用者不用持有具体操作者的引用,而是通过代理者去对具体操作者执行具体的操作。

静态代理的实现

操作接口:

public interface Operate {    voiddoSomething();}

操作者:

public class Operator implements Operate {    @Override    public voiddoSomething() {        System.out.println("I'm doing something");    }}

代理者:

public class OperationProxy implements Operate {    private Operator operator = null;    @Override    public voiddoSomething() {        beforeDoSomething();if(operator == null){            operator =  new Operator();        }        operator.doSomething();        afterDoSomething();    }    private voidbeforeDoSomething() {        System.out.println("before doing something");    }    private voidafterDoSomething() {        System.out.println("after doing something");    }}

调用者:

public class StaticProxyTest {    public static void main(String[] args) {        Operate operate = new OperationProxy();//使用OperationProxy代替Operator        operate.doSomething();  //代理者代替真实者做事情    }}

静态代理的局限性

可以看到,静态代理让调用者不用再直接持有操作者的引用,而是将一切操作交由代理者去完成。但是静态代理也有它的局限性:

如果需要增加一个需要代理的方法,代理者的代码也必须改动进而适配新的操作;

如果需要代理者代理另外一个操作者,同样需要对代理者进行扩展并且更加麻烦。

可能有人想到可以用策略模式和工厂模式分别解决上面两个问题,但是,有没有更加巧妙的方法呢?首先,我们了解一下 Java 代码的执行过程。

理解 Java 代码执行流程

要从根本上理解动态代理的实现原理,得先从 Java 代码的执行流程说起:

JVM 在运行 .class 文件之前,首先通过 ClassLoader 将 .class 文件以二进制的形式解析并生成实例以供调用,我们的代码执行逻辑是在 JVM 的运行期系统中进行工作的,那么,我们可不可以在自己的代码里面按照 .class 的格式生成自己的 .class 文件,进而调用自定义的 ClassLoader 将其加载出来呢?答案是肯定的,这样我们就可以动态地创建一个类了。

生成自己的 .class 文件

当然我们不用手动去一点一点拼装 .class 文件,目前比较常用的字节码生成工具有ASMJavassist,根据这个思路,生成 .class 文件的过程如下:

import javassist.ClassPool;import javassist.CtClass;import javassist.CtMethod;import javassist.CtNewMethod; public class Test {    public static void main(String[] args) throws Exception {        ClassPool pool = ClassPool.getDefault();        //创建 AutoGenerateClass 类        CtClass cc= pool.makeClass("com.guanpj.AutoGenerateClass");        //定义 show 方法        CtMethod method = CtNewMethod.make("public void show(){}", cc);        //插入方法代码        method.insertBefore("System.out.println(\"I'm just test generate .class file by javassit.....\");");        cc.addMethod(method);        //保存生成的字节码        cc.writeFile("D://temp");    }}

生成的 .class 文件如下:import javassist.ClassPool;import javassist.CtClass;import javassist.CtMethod;import javassist.CtNewMethod; public class Test { public static void main(String[] args) throws Exception { ClassPool pool = ClassPool.getDefault(); //创建 AutoGenerateClass 类 CtClass cc= pool.makeClass("com.guanpj.AutoGenerateClass"); //定义 show 方法 CtMethod method = CtNewMethod.make("public void show(){}", cc); //插入方法代码 method.insertBefore("System.out.println(\"I'm just test generate .class file by javassit.....\");"); cc.addMethod(method); //保存生成的字节码 cc.writeFile("D://temp"); }}

反编译后查看内容:

//// Source code recreated from a .class file by IntelliJ IDEA// (powered by Fernflower decompiler)//package com.guanpj;public class AutoGenerateClass {    public voidshow() {        System.out.println("I'm just test generate .class file by javassit.....");    }    publicAutoGenerateClass() {    }}

可以看到,javassit 生成的类中,除了 show() 方法之外还默认生成了一个无参的构造方法。

自定义类加载器加载

为了能够让自定的类被加载出来,我们自定义了一个类加载器来加载指定的 .class 文件:

public class CustomClassLoader extends ClassLoader {    publicCustomClassLoader() {    }    protected Class findClass(String className) {        String path ="D://temp//"+ className.replace(".","//") +".class";        byte[] classData = getClassData(path);returndefineClass(className, classData, 0, classData.length);    }    private byte[] getClassData(String path) {        try {            InputStream ins = new FileInputStream(path);            ByteArrayOutputStream baos = new ByteArrayOutputStream();            int bufferSize = 4096;            byte[] buffer = new byte[bufferSize];            int bytesNumRead = 0;while((bytesNumRead = ins.read(buffer)) != -1) {                baos.write(buffer, 0, bytesNumRead);            }returnbaos.toByteArray();        } catch (IOException e) {            e.printStackTrace();        }returnnull;    }}

接着,用 ClassLoader 加载刚才生成的 .class 文件:

public class TestLoadClass {    public static void main(String[] args) throws Exception {        CustomClassLoader classLoader = new CustomClassLoader();        Class clazz = classLoader.findClass("com.guanpj.AutoGenerateClass");        Object object = clazz.newInstance();        Method showMethod = clazz.getMethod("show", null);        showMethod.invoke(object, null);    }}

后台输出如下:

成功执行了 show 方法!

利用 JDK 中的 Proxy 类进行动态代理

使用动态代理的初衷是简化代码,不管是 ASM 还是 Javassist,在进行动态代理的时候操作还是不够简便,这也违背了我们的初衷。我们来看一下怎么 InvocationHandler 怎么做:

InvocationHandler:

public class InvocationHandlerImpl implements InvocationHandler {    Operate operate;    //注入操作者对象    public InvocationHandlerImpl(Operate operate) {        this.operate = operate;    }    @Override    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {        System.out.println("before calling method: "+ method.getName());        //调用操纵者的具体操作方法        method.invoke(operate, args);        System.out.println("after calling method: "+ method.getName());returnnull;    }}

调用者:

public class DynamicProxyTest {    public static void main(String[] args) {        //实例化操作者        Operate operate = new Operator();        //将操作者对象进行注入        InvocationHandlerImpl handler = new InvocationHandlerImpl(operate);        //生成代理对象        Operate operationProxy = (Operate) Proxy.newProxyInstance(operate.getClass().getClassLoader(),                operate.getClass().getInterfaces(), handler);        //调用操作方法        operationProxy.doSomething();    }}

跟静态代理不同的是,动态代理的过程主要分为三个步骤

将操作者对象注入 InvocationHandlerImpl 类中。

将 InvocationHandlerImpl 对象注入 Proxy 类中并返回代理者对象,并在 invoke 方法中进行额外的操作

调用代理对象的操作方法

利用 CGLIB 进行动态代理

用 Proxy 类生成代理类的方法为 newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) ,第二个参数是操作者的接口数组,意味着只能代理它实现的接口里的方法,对于本来在操作者类中定义的方法表示无能为力,CGLIB(Code Generation Library) 解决了这个问题。

MethodInterceptorImpl:

public class MethodInterceptorImpl implements MethodInterceptor {    @Override    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {        System.out.println("before calling method:"+ method.getName());        proxy.invokeSuper(obj, args);        System.out.println("after calling method:"+ method.getName());returnnull;    }}

调用者:

public class ProxyTest {    public static void main(String[] args) {        Operator operator = new Operator();        MethodInterceptorImpl methodInterceptorImpl = new MethodInterceptorImpl();        //初始化加强器对象        Enhancer enhancer = new Enhancer();        //设置代理类        enhancer.setSuperclass(operator.getClass());        //设置代理回调        enhancer.setCallback(methodInterceptorImpl);        //创建代理对象        Operator operationProxy = (Operator) enhancer.create();        //调用操作方法        operationProxy.doSomething();    }}

使用 CGLIB 进行动态代理的过程分为四个步骤:

使用 MethodInterceptorImpl 实现 MethodInterceptor 接口,并在 intercept 方法中进行额外的操作

创建增强器 Enhance 并设置被代理的操作类

生成代理类

调用代理对象的操作方法

总结

无论是静态代理还是动态代理,都能一定程度地解决我们的问题,在开发过程中可以根据实际情况选择合适的方案。总之,没有好不好的方案,只有适不适合自己项目的方案,我们应该深入研究和理解方案背后的原理,以便能够应对开发过程中产生的变数。

在此我向大家推荐一个架构学习交流群。交流学习群号:938837867 暗号:555 里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备

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

推荐阅读更多精彩内容