代理模式,以及Java的动态代理

定义

为其他对象提供一种代理以控制对这个对象的访问。可以提供额外不同的操作。

UML类图

Subject类:定义了RealSubject和Proxy的共用接口,这样就在任何使用RealSubject的地方都可以使用Proxy

RealSubject类:真是实体

Proxy类:代理类,保存一个引用使得代理可以访问实体,并提供一个与Subject接口相同的接口,这样代理就可以用来替代实体。

简单的代码类如下:

public abstract class Subject {
    public abstract void Request();
}

public class RealSubject extends Subject {
    @Override public void Request() {
        System.out.println("真实请求");
    }
}

public class Proxy extends Subject {
    private RealSubject realObject = new RealSubject();
    @Override public void Request() {
        System.out.println("代理的请求");
        realObject.Request();
    }
}

public class Client {
    public static void main(String[] args) {
        Subject proxy = new Proxy();
        proxy.Request();
    }
}

示例

举个栗子,大学的时候经常在宿舍里不想出去,有时候让室友帮忙在食堂里带饭。

我:在宿舍不想出去

室友:帮我带饭

public abstract class Subject {
    public abstract void buyLaunch();
}

public class RealSubject extends Subject {
    @Override public void buyLaunch() {
        System.out.println("我要买午饭");
    }
}

public class Proxy extends Subject {
    private RealSubject realObject = new RealSubject();

    @Override public void buyLaunch() {
        System.out.println("室友帮我买午饭");
        wrapLaunch();
        realObject.buyLaunch();
    }

    private void wrapLaunch() {
        System.out.println("打包带走");
    }
}

代理模式其实就是这么简单~~~

 

Java的动态代理

接下来看一下Java提供的动态代理,动态代理比代理的思想更前进了一步,因为它可以动态地创建代理并动态地处理对所有方法的调用。在动态代理上所做的所有调用都会被重定向到单一的调用处理器上,它的工作是揭示调用的类型并确定相应的对策。

先来看一下动态代理的例子:

public interface ObjectInterface {
    void doSomething();

    void somethingElse(String arg);
}

public class RealObject implements ObjectInterface {
    @Override public void doSomething() {
        System.out.println("doSomething");
    }

    @Override public void somethingElse(String arg) {
        System.out.println("somethingElse " + arg);
    }
}
public class DynamicProxyHandler implements InvocationHandler {
    private Object proxied;

    public DynamicProxyHandler(Object proxied) {
        this.proxied = proxied;
    }

    @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("Proxy: " + proxy.getClass() + ", method:" + method + ", args:" + args);
        if (args != null) {
            for (Object arg : args) {
                System.out.println("  " + arg);
            }
        }
        return method.invoke(proxied, args);
    }
}
public class SimpleDynamicProxy {

    public static void consumer(ObjectInterface iface) {
        iface.doSomething();
        iface.somethingElse("bonobo");
    }

    public static void main(String[] args) {
        RealObject real = new RealObject();
        ObjectInterface proxy = (ObjectInterface) Proxy.newProxyInstance(ObjectInterface.class.getClassLoader(),
                new Class[]{ObjectInterface.class}, new DynamicProxyHandler(real));
        consumer(proxy);
    }
}

通过调用静态方法Proxy.newProcyInstance()创建动态代理,这个方法需要得到一个类加载器,一个你希望该代理实现的接口列表,以及InvocationHandler接口的一个实现,然后返回接口类型的对象。动态代理可以将所有调用重定向到调用处理器,因此通常会向调用处理器的构造器传递给一个“实际”对象的引用,从而使得调用处理器在执行其中介任务时,可以将请求转发。

看一下SimpleDynamicProxy运行的输入内容:

在consumer方法中,调用ObjectInterface的任何方法,都会先经过DynamicProxyHandler的invoke方法中转,实现统一处理的功能。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值