动态代理讲解(一)之反射≈复用?

2018-07-22  本文已影响0人  维特无忧堡

1、介绍

  在讲解动态代理之前,我们要弄懂一个概念:
  什么是代理模式?

可以看成是对调用目标的一个封装,我们对目标代码的调用不是解法发生的,而是通过代理完成的,这在一定程度上解决了调用者和实现者这之间的耦合。

 什么意思呢?
 举个例子吧,比如说我们要租房,最省事的办法就是找个中介,你只要告诉他你需要什么型号的房,然后啥都不用管,他会帮你找房,非常省事,他会帮你找合适的房东,你最后只要交钱就行了,那么在这里,房东就是调用目标,而中介就是我们所说的代理者,你可以根据代理者而间接达到你租房的目的。那么这个场景用代码怎么实现呢?

public interface Room {
    void rent();
}
class RoomImpl implements Room{
    @Override
    public void rent() {
        System.out.println("三室一厅");
    }
}
class RoomProxy implements Room{
    private Room room;
    public RoomProxy(Room room){
        this.room = room;
    }
    @Override
    public void rent() {
        System.out.println("租房前准备");
        room.rent();
        System.out.println("租完了");
    }
}
public class Client {
    public static void main(String[] args) {
        RoomImpl impl = new RoomImpl();
        RoomProxy proxy = new RoomProxy(impl);
        proxy.rent();
    }
}
image.png

这样你就应该有个比较直观的感觉了吧?那么你想一下这样写有没有什么不足?想一想,如果我一个类中所代理的方法有很多,这样是不是要无形之中增加很多方法?还有,像刚刚这个例子,每次租房前都要准备合同啊等各种必要工作,那是不是会产生代码冗余呢?而且维护起来也不方便,那么有什么办法能解决这样一个问题呢?

哎!这时候动态代理就出来了,利用反射机制,可以使得相应的调用操作动态完成。看运行时调用哪个就代理哪个,这样就只需要一个方法就可以了,这样是不是就实现了复用了呢?

提供一下代码瞧瞧吧!这样比较直观:

public class DynaProxy {

    public static void main(String[] args) {
        Work work = new WorkImpl();
        MyInvocationHandle handle = new MyInvocationHandle(work);

        Work proxy = (Work) Proxy.newProxyInstance(WorkImpl.class.getClassLoader(),WorkImpl.class.getInterfaces(),
                                             handle);
        proxy.dowork();
    }
}

interface Work{
    void dowork();
}

class WorkImpl implements Work{

    @Override
    public void dowork() {
        System.out.println("do work");
    }
} 
class MyInvocationHandle implements InvocationHandler{

    private Object object;//被代理对象

    public MyInvocationHandle(Object object) {
        this.object = object;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("befor");
        Object result = method.invoke(object,args);
        System.out.println("after");
        return result;
    }
}

首先我们实现InvocationHandler,以接口为纽带构建Proxy代理对象,通过对代理对象的调用间接实现对目标对象的调用

newProxyInstance返回的是一个实现了dowork接口的一个代理类。

上一篇 下一篇

猜你喜欢

热点阅读