动态代理(JDK动态代理剖析)

动态代理

代理模式

概念:为其它对象提供一种代理以控制对这个对象的访问
本质:触发被代理安排;但是执行者还是被代理本身执行(包括动态代理)

类图

代理模式UML.png
  • Subject抽象主题角色

    可以是抽象类也可以是接口

  • RealSubject具体主题角色

    委托角色、被代理角色(业务逻辑的具体执行者)

  • Proxy代理主题角色

    委托类、代理类;它负责对真实角色的应用,把所有抽象主题类定义的方法委托给真实主题角色实现,并且在真实主题角色处理完毕前后做预处理和善后处理的工作。

    抽象主题类

    public interface Subject {
        void method();
    }
    

    真实主题类

    public class RealSubject implements Subject {
        //实现方法
        @Override
        public void method() {
            //具体业务处理
        }
    }
    

    代理类

    public class Proxy implements Subject {
        //要代理的对象
        private Subject subject;
        //构造函数中;传入要代理的对象
        public Proxy(Subject realSubject) {
            this.subject = realSubject;
        }
        //实现接口中定义的方法
        @Override
        public void method() {
            //预处理
            this.before();
            //代理的对象执行业务逻辑
            this.subject.method();
            //善后处理
            this.after();
        }
        private void before() {
            //预处理
        }
        private void after() {
            //善后处理
        }
    }
    

为什么使用代理

  • 职责清晰

    真实的角色就是实现实际的业务逻辑,不用关心其它非本职的工作

  • 高扩展

  • 补充功能

动态代理

概念

在实现阶段不关心代理谁,而在运行阶段才指定代理哪个对象

为什么会产生动态代理

  • 静态代理代理类和被代理类实现了相同的接口,导致代码重复,可扩展性差(被代理类增加方法,代理类也要实现这个方法)
  • 静态代理代理类只服务于一种类型的代理

动态代理分类

动态代理主要分为JDK动态代理和cglib动态代理两大类

JDK动态代理

必要条件:必须有被代理类的接口

UML图

动态代理UML.png

抽象被代理角色

public interface Subject {
    //业务操作
    public void doSomething(String str);
}

真实被代理角色

public class RealSubject implements Subject{
    //业务操作
    @Override
    public void doSomething(String str) {
        System.out.println("do something------> " + str);
    }
}

动态代理Handler

public class MyInvocationHandler implements InvocationHandler{
    //被代理的对象
    private Object target;
    //通过构造函数传入被代理对象
    public MyInvocationHandler(Object target) {
        this.target = target;
    }
    //参数1:proxy 动态代理对象
    //参数2:method 正在执行的方法
    //参数3:args 调用目标方法时传入的实参
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) 
            throws Throwable {
        //执行被代理的方法
        return method.invoke(this.target, args);
    }

测试类1

public class Client1 {
    public static void main(String[] args) {
        //定义被代理类
        Subject realSubject = new RealSubject();
        //定义一个handler
        InvocationHandler handler = new MyInvocationHandler(realSubject);
        //动态代理类
        //第一个参数:用于定义代理类的类加载器;传入被代理类接口的类加载器(动态)
         //注意 第一个参数加载器 不是handler的加载器,handler不是代理类
        //第二个参数:被代理类的接口
        //第三个参数:InvocationHandler,用来处理方法的调用
        Subject proxy = (Subject) Proxy.newProxyInstance(
                realSubject.getClass().getClassLoader(), 
                realSubject.getClass().getInterfaces(), 
                handler);
        //代理的行为
        proxy.doSomething("Finish");
    }

}

输出:

do something------> Finish

测试类2

public class Client2 {
    public static void main(String[] args) 
            throws Exception {
        //定义被代理类
        Subject realSubject = new RealSubject();
        //定义一个handler
        InvocationHandler handler = new MyInvocationHandler(realSubject);
        //使用Proxy生成一个动态代理类ProxyClass
        Class<?> proxyClass = Proxy.getProxyClass(Subject.class.getClassLoader(), 
                new Class[] {Subject.class});
        //获取proxyClass类中一个invocationHandler参数的构造器
        Constructor<?> constructor = 
                proxyClass.getConstructor(new Class[] {InvocationHandler.class});
        //调用constructor 的newInstance方法创建动态实例
        Subject proxy = (Subject) constructor.newInstance(new Object[] {handler});
        //代理的行为
        proxy.doSomething("Finish");
    }
}

输出:

do something------> Finish

JDK动态代理分析

Proxy

提供用于创建动态代理类和代理对象的静态方法,它也是所有动态代理类的父类。

Proxy提供了如下两个方法来创建动态代理类和动态代理实例

public static Class<?> getProxyClass(ClassLoader loader,
                                         Class<?>... interfaces)    

创建一个动态代理类所对应的class对象,该代理类将实现interfaces所指定的所有接口。如以上测试类2

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

直接创建一个动态代理对象,该代理对象实现了interfaces指定的系列接口,执行代理对象的每个方法时都会被替换执行InvocationHandler对象的invoke方法

动态代理类由输出到硬盘

public class CreateProxyClass {
    public static void main(String[] args) {
        // 使用Proxy生成一个动态代理类ProxyClass
        Class<?> proxyClass = Proxy.getProxyClass(
                Subject.class.getClassLoader(), 
                new Class[] { Subject.class });
        //将动态生成的代理类生成字节码;并输出到D盘Proxy.class
        byte[] classFile = ProxyGenerator.generateProxyClass("$Proxy", 
                proxyClass.getInterfaces());
        String path = "D://Proxy.class";
        try (FileOutputStream out = new FileOutputStream(path);){
            out.write(classFile);
            out.flush();
        } catch (Exception e) {
            e.printStackTrace();
        } 
    }
}

查看 D://Proxy.class

import com.dwb.design.proxy.demo1.*;
import java.lang.reflect.*;

public final class $Proxy extends Proxy implements Subject
{
    private static Method m1;
    private static Method m2;
    private static Method m3;
    private static Method m0;
    
    public $Proxy(final InvocationHandler invocationHandler) {
        super(invocationHandler);
    }
    
    public final boolean equals(final Object o) {
        try {
            return (boolean)super.h.invoke(this, $Proxy.m1, new Object[] { o });
        }
        catch (Error | RuntimeException error) {
            throw;
        }
        catch (Throwable t) {
            throw new UndeclaredThrowableException(t);
        }
    }
    
    public final String toString() {
        try {
            return (String)super.h.invoke(this, $Proxy.m2, null);
        }
        catch (Error | RuntimeException error) {
            throw;
        }
        catch (Throwable t) {
            throw new UndeclaredThrowableException(t);
        }
    }
    
    //重点查看该方法:我们写的业务方法;
    public final void doSomething(final String s) {
        try {
            //对应到 InvocationHandler的三个参数;为什么第一个为动态代理类
            //解释了,调用方法最后都会调用重写的InvocationHandler 的invoke方法
            super.h.invoke(this, $Proxy.m3, new Object[] { s });
        }
        catch (Error | RuntimeException error) {
            throw;
        }
        catch (Throwable t) {
            throw new UndeclaredThrowableException(t);
        }
    }
    
    public final int hashCode() {
        try {
            return (int)super.h.invoke(this, $Proxy.m0, null);
        }
        catch (Error | RuntimeException error) {
            throw;
        }
        catch (Throwable t) {
            throw new UndeclaredThrowableException(t);
        }
    }
    
    static {
        try {
            $Proxy.m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
            $Proxy.m2 = Class.forName("java.lang.Object").getMethod("toString", (Class<?>[])new Class[0]);
            $Proxy.m3 = Class.forName("com.dwb.design.proxy.demo1.Subject").getMethod("doSomething", Class.forName("java.lang.String"));
            $Proxy.m0 = Class.forName("java.lang.Object").getMethod("hashCode", (Class<?>[])new Class[0]);
        }
        catch (NoSuchMethodException ex) {
            throw new NoSuchMethodError(ex.getMessage());
        }
        catch (ClassNotFoundException ex2) {
            throw new NoClassDefFoundError(ex2.getMessage());
        }
    }
}

运行时动态代理UML

运行时动态代理UML.png

说明:蓝色部分为运行时生成的代理对象$Proxy; 该代理类继承了Proxy、实现了被代理接口、组合了InvocationHandler;

代理对象执行过程:

  1. 先调用自身实现的被代理接口方法
  2. 调用组合的invocation对象,invoke方法(参数1:代理对象,参数2:被代理接口方法,参数3:方法参数)
  3. 反射执行被代理方法;invocation对象中invoke方法体内method.invoke(this.target, args)

未完(CGLIB)

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

推荐阅读更多精彩内容