代理模式

为其他对象提供一个代理以控制对这个对象的访问,在不改变目标对象基础上添加额外的功能。

如何理解代理模式

  • 两个主体:代理对象和被代理对象
  • 对于被代理对象,事情必须要去做吗,但是自己不想去做或是没有条件去做,需要代理对象代替被代理对象去做。
  • 代理对象能够获取被代理对象的资料信息。 代理层面(获得被代理对象的引用)

生活案例

  • 中介:通常买二手车的时候,会去网上找车源,对车进行质量检测,以及各种过户手续的办理,自己可能不想做这些事情,所以可以找第三方的中介公司来完成这些事情,我只负责把我想要的车辆的信息(价位、车辆新旧程度、品牌)反馈给中介,中介把所有的办好就只管我来签字验收就可以了。

  • 黄牛:当春运火车票比较紧张的时候,我抢不到票,且自己也不想去抢,则可以通过黄牛去买,我只管把我的车票信息给他(初始地、目的地、出发时间、车次),由此不用抢票,也可以买到相应的车票,抢票的过程由黄牛去做。

  • 媒人:平时自己没有时间去交女朋友,则需要媒人去介绍,我们只管把我们想要的女朋友的类型给媒人,媒人则帮我们去寻找合适的对象。

需要类图

  • Subject(抽象主题)

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

  • RealSubject(具体角色)

被代理的对象,业务逻辑的执行者

  • Proxy(代理角色)

代理类,负责对真实角色的应用,把所有抽象主题定义的方法限制委托给真实角色实现,并在真实角色处理完毕前后做相应工作

几种代理实现形式

代理有两种主要形式

  1. 静态代理

编译前,代码已经创建好,程序运行时,class文件已经存在

  1. 动态代理

程序运行时通过反射生成代理类

静态代理

  • 代理类与目标类实现相同的接口
  • 代理类持有目标类的引用,可以控制目标类方法的访问
普通代理
public interface Subject {

    void request();
}
public class RealSubject implements Subject {
    @Override
    public void request() {
        System.out.println("start to request");
    }
}
public class Proxy implements Subject{
    private Subject subject;

    public Proxy(Subject subject){
        this.subject = subject;
    }

    @Override
    public void request() {
        this.before();
        subject.request();
        this.after();
    }

    private void before(){
        System.out.println("前置处理。。。。");
    }

    private void after(){
        System.out.println("后置处理。。。。");
    }
}
强制代理
  • 只能通过真实角色角色制定的代理类才能访问
  • 不允许直接访问真实角色
public interface Subject {
    void request();

    Subject getProxy();
}
```java
public class SubjectProxy implements Subject{
    private Subject subject;

    public SubjectProxy(Subject subject) {
        this.subject = subject;
    }

    @Override
    public void request() {
        this.subject.request();
    }

    @Override
    public Subject getProxy() {
        return this;
    }
}
```java
public class RealSubject implements Subject{
    private Subject proxy = null;
    @Override
    public void request() {
        if(this.isProxy()){
            System.out.println("start to request");
        }else {
            System.out.println("请使用指定的代理进行访问");
        }

    }

    @Override
    public Subject getProxy() {
        this.proxy = new SubjectProxy(this);
        return this.proxy;
    }

    private boolean isProxy(){
        if(this.proxy == null){
            return false;
        }else {
            return true;
        }
    }
}
public class Client {
    public static void main(String[] args) {
        RealSubject subject = new RealSubject();
        Subject proxy = subject.getProxy();
        proxy.request();
    }
}
静态代理局限性
  • 不够灵活,每个代理只为一个接口服务,会造成大量代码重复
  • 静态代理类职位特定的接口服务,如果有多个接口,需要生成多个代理对象,不利于系统维护

动态代理

public interface CustomerService {
    void buyCar();
}
public class CustomerServiceImpl implements CustomerService{
    @Override
    public void buyCar() {
        System.out.println("决定买这个车了");
    }
}
public class CarAgency implements InvocationHandler{
    //被代理对象的引用
    private Object target;

    public Object getInstance(Object target){
        this.target = target;
        Class clazz = target.getClass();
        return Proxy.newProxyInstance(clazz.getClassLoader(),target.getClass().getInterfaces(),this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("办理车辆质量检查报告");
        method.invoke(target,args);
        System.out.println("办理车辆二手车过户");
        return null;
    }
}
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容