码迷,mamicode.com
首页 > 其他好文 > 详细

十七、代理设计模式

时间:2016-05-13 03:43:25      阅读:139      评论:0      收藏:0      [点我收藏+]

标签:

1. 代理设计模式介绍

代理设计模式,从名字中我们就能知道其大致意思。我们生活中有很多“代理”,比如,租房中介,海外代购等。就是一种本来需要A做的事,让B去代做。

定义:

为其他对象提供一种代理以控制对这个对象的访问。

2. 代理设计模式使用场景

  • 当我们无法直接访问某一个对象时,可以通过一个代理对象间接访问。通常委托对象和代理对象有着相同的接口。

  • (1)、远程代理,为一个对象在不同的地址空间提供局部代表,这样可以隐藏一个对象存在于不同地址空间的事实。

  • (2)、虚拟代理,是根据需要创建开销很大的对象,通过它来存放实例化需要很长时间的真实对象。
  • (3)、安全代理,用来控制真实对象访问时的权限。
  • (4)、智能引用,当调用真实对象时,代理处理另外一些事。

3. 代理设计模式UML类图

技术分享

角色介绍:

  • Subject: 抽象主题类,该类的主要职责是声明真实主题与代理的共同接口方法,该类既可以是一个抽象类,也可以是一个接口。

  • RealSubject:真实主题,该类定义类代理所表示的真实对象,由其执行具体的业务逻辑方法。

  • ProxySubject: 代理类,该类持有一个对真实主题的引用,在其所实现的接口方法中调用真实主题对应接口中的方法。

  • Client:测试类

4. 代理设计模式简单实现

  • (1)、抽象主题类:
public interface Subject {
    void visit();
}

抽象主题类中,只有一个visit()方法,代理对象和被代理代理对象均要实现这个方法。

  • (2)、真实主题:
public class RealSubject implements Subject {
    @Override
    public void visit() {
        System.out.println("Real Subject");
    }
}

真实对象中,继承自抽象主题类,在visit()方法中,实现具体逻辑。

  • (3)、代理主题:
public class ProxySubject implements Subject {
    private RealSubject mSubject;

    public ProxySubject(RealSubject mSubject) {
        this.mSubject = mSubject;
    }

    @Override
    public void visit() {
        //通过真实主题引用的对象调用真实主题中的逻辑方法
        mSubject.visit();
    }
}

ProxySubject代理类中有一个真实主题类的引用,同时代理主题类继承了和真实主题类一模一样的抽象类,实现了同样的方法。只不过在代理类中,visit()方法调用了真实主题类的visit()方法。

  • (4)、测试类:
public class Client {

    public static void main(String[] args) {
        //构建一个真实的主题对象
        RealSubject realSubject = new RealSubject();

        //通过一个真实的主题对象构造一个代理对象
        ProxySubject proxySubject = new ProxySubject(realSubject);

        //调用代理的相关方法
        proxySubject.visit();
    }
}

以上模式叫做静态代理,还有一种叫做动态代理模式。

静态代理模式就如上述所示,我们的代码在运行前类的class编译文件就已经存在;而动态代理类则与静态代理类相反,通过反射机制动态地生成代理者对象,代理对象将会在执行阶段执行。

方法如下:我们声明一个动态代理类:

public class DynamicProxy implements InvocationHandler {
    private Object object;//被代理对象的引用

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

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

在上述代码中,我们实现了动态代理接口InvocationHandler,这是java为我们提供的,我们只需要重写其调用方法invoke即可。

在以上代码中,我们声明了一个Object的引用,该引用将指向被代理类,而我们在invoke方法中调用具体的被代理方法,也就是真实方法。

接着我们修改Client类方法如下:

public class Client {

    public static void main(String[] args) {
        RealSubject realSubject = new RealSubject();
        //构造一个动态代理对象
        DynamicProxy dynamicProxy = new DynamicProxy(realSubject);
        //类加载器
        ClassLoader classLoader = dynamicProxy.getClass().getClassLoader();
        Subject subject = (Subject) Proxy.newProxyInstance(classLoader, new Class[]{Subject.class}, dynamicProxy);

        //调用代理对象的visit方法。
        subject.visit();
    }
}

5. 总结

  • 代理设计模式应用比较广泛,缺点较少。

十七、代理设计模式

标签:

原文地址:http://blog.csdn.net/u010649376/article/details/51340014

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!