Java动态代理

软件开发的最终目的是以不变应万变,为此我们的前辈们总结了很多模式来应对各种需求,代理模式就是其中的一种。假如我们已经理解了静态代理,那动态代理就相对容易理解了,被代理的类是动态变化的,是一个泛型的 target; 那到底怎么实现的呢。
Java的动态代理,会用到InvocationHandler、Proxy 。代码设计的时候考虑到方便扩展和维护,都需要抽象出公共特性,例如共有的方法。 动态代理的出现是为了解决:
1.控制外部调用
2.增强某些方法的处理
3.需要在N多类的同一个方法执行前或者执行后做一些事情(AOP)

代码例子:

/**
 * 公共接口,抽象出公共方法
 */
public interface Person {
    public void giveTask();
}

/**
 * 具体的人,例如学生,实现交作业的方法
 */
public class Student implements  Person {
    private String name;
    public  Student(String name) {
        this.name = name;
    }

    public void task() {
        System.out.println(name+"交作业");
    }
}

/**
 * 具体的人,老师改作业
 */
public class Tescher implements Person {

    @Override
    public void task() {
        System.out.println("老师改作业");
    }
}

如果不用代理,那直接就调用Student和Teacher类了,那我们现在想在所有的task方法执行前都干点事情,也许你说可以用抽象类,然后抽象类中实现方法进行处理,也不是不行,但是这样带来问题,我将来想扩展更多的类是不是需要改动父类或者我super调用不及时,等等这样的问题。 所以动态代理的出现就是为了不侵入的方式去增强原有的方法。

/**
 * 我想在原有的老师或者学生做事情前,干点啥又不用去改动原来的方法
 * 这就是一个Person代理方法处理,并不是代理类哦,代理是通过他动态创建的
 * 在代理类里调用本类的invoke方法,从而调用目标类的方法
 * @param <T> 泛型
 */
public class PersonInvocationHandler<T> implements InvocationHandler {
    // 被代理的对象
    T target;

    public PersonInvocationHandler(T target) {
        this.target = target;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("代理执行" + method.getName() +"方法");
        // 用反射执行被代理对象的方法
        Object ret = method.invoke(target,args);
        return ret;
    }
}

具体调用:

    public static void main(String[] args) {
        // 学生
        Person xiaoming = new Student("小明");
        // 老师
        Person laoshi = new Tescher();

        // 构建代理方法处理者,就是通过他要在别人方法前后干点事情
        PersonInvocationHandler invocationHandler1 = new PersonInvocationHandler(laoshi);

        // 通过Proxy.newProxyInstance 生成一个新的代理类对象,这个方法硬生生的创造出了一个新的类出来$Proxy0
        // 并且按照定义的接口,实现了接口中的方法,方法内部是调用 invocationHandler的invoke()
        Person person1 = (Person)Proxy.newProxyInstance(Tescher.class.getClassLoader(),new Class<?>[]{Person.class},invocationHandler1);

        // 表面上看是调用Person.task(), 实际上是调用新创建的代理类的task()方法,神不知鬼不觉。
        person1.task();

    }

运行结果如你猜想的一样:

代理执行task方法
老师改作业

动态代理之所以动态,就在于用泛型接收被代理对象和完全动态的创建了一个新的代理类出来。 怎么知道创建了新的类呢,在Proxy.newProxyInstance前设置一下参数,就会输出动态创建的类到项目目录下,$Proxy0.class这个就是刚刚创建被编译成字节码的文件。

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

利用IDEA反编译查看一下

/*
继承Proxy,方便调用invocationHandler,还记得Proxy.newProxyInstance传进去的,实现Person接口
*/
public final class $Proxy0 extends Proxy implements Person {
//以下是提取的 方法
    private static Method m1;
    private static Method m2;
    private static Method m3;
    private static Method m0;
     // 其他部分省略了,这里是获取方法,利用反射找到被代理的方法
    static {
        try {
            m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
            m2 = Class.forName("java.lang.Object").getMethod("toString");
            m3 = Class.forName("com.alienjun.dynamictest.Person").getMethod("task");
            m0 = Class.forName("java.lang.Object").getMethod("hashCode");
        } catch (NoSuchMethodException var2) {
            throw new NoSuchMethodError(var2.getMessage());
        } catch (ClassNotFoundException var3) {
            throw new NoClassDefFoundError(var3.getMessage());
        }
    }

    // 根据接口实现task()方法,这里只是调用父类的invocationHandler.invoke(),具体执行task是在invocationHandler.invoke中执行的
    public final void task() throws  {
        try {
            super.h.invoke(this, m3, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }
}

到这里就已经明白整个动态代理过程了,
动态代理

其中怎么创建的类,又是怎么被编译和加载调用的更多细节可以细看源码。

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