美文网首页
代理模式

代理模式

作者: wervy | 来源:发表于2020-01-29 11:30 被阅读0次

定义

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

使用场景

当无法或不想直接访问某个对象,或者访问某个对象困难时,可以通过某个代理对象间接访问。为了保证客户端使用的透明性,委托对象与代理对象需要实现相同的接口。

角色介绍

Subject:抽象主题类
该类的主要职责是声明真实主题与代理的共同接口方法,该类既可以一个抽象类,也可以是一个抽象接口
RealSubject:真实主题类
该类也称为被委托类或被代理类,该类定义了代理所表示的真实对象,由其执行具体的业务逻辑方法, 而客户类则通过代理类间接地调用真实主题类中定义的方法。
ProxySubject:代理类
该类也称为委托类或代理类,该类持有一个对真实主题类的引用,在其所实现的接口方法中调用真实主题类中相应的接口方法执行,以此起到代理的作用
Client:客户类,即使用代理类的类型

示例代码:

//抽象主题
public interface ILawsuit {

    //提交申请
    void submit();
    //进行举证
    void burden();
    //开始辩护
    void defend();
    //诉讼完成
    void finish();
}

//真实主题类
public class XiaoMin implements ILawsuit{



    @Override
    public void submit() {

        System.out.println("老板拖欠工资,特此申请仲裁!");

    }

    @Override
    public void burden() {

        System.out.println("这是合同书和过去一年的银行工资流水!");
    }

    @Override
    public void defend() {

        System.out.println("证据确凿,不需要再说什么了!");
    }

    @Override
    public void finish() {

        System.out.println("诉讼成功,判决老板即日起七天结算工资");
    }
}
//代理类
public class Lawyer implements ILawsuit{

    private ILawsuit mLawsuit;//持有一个具体被代理者的引用

    public Lawyer(ILawsuit lawsuit){

        mLawsuit = lawsuit;
    }

    @Override
    public void submit() {

        mLawsuit.submit();

    }

    @Override
    public void burden() {
       mLawsuit.burden();
    }

    @Override
    public void defend() {
    mLawsuit.defend();
    }

    @Override
    public void finish() {
        mLawsuit.finish();
    }
}

//客户类
public class Client {

    public static void main(String[] args){

        //构造一个小民
        ILawsuit xiaomin = new XiaoMin();
        //构造一个代理律师,并将小民作为构造参数传递进去
        ILawsuit lawyer = new Lawyer(xiaomin);
        //律师提交诉讼申请
        lawyer.submit();
        //律师进行举证
        lawyer.burden();
        //律师代替小民进行辩护
        lawyer.defend();
        //完成诉讼
        lawyer.finish();

    }
}

代理模式分为两种:

  • 静态代理:代码运行前代理类的class编译文件就已经存在
  • 动态代理:通过反射机制动态地生成代理者的对象,也就是我们在code阶段压根就不需要知道代理谁,代理谁我们将会在执行阶段决定,而Java也给我们提供了一个便捷的动态代理接口InvocationHandler,实现该接口需要重写其调用方法invoke.
public class DynamicProxy implements InvocationHandler{

    private Object obj; //被代理的类引用

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

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        Object result = method.invoke(obj,args);

        return result;
    }
}

import java.lang.reflect.Proxy;

public class Client {

    public static void main(String[] args){

        //构造一个小民
        ILawsuit xiaomin = new XiaoMin();
//        //构造一个代理律师,并将小民作为构造参数传递进去
//        ILawsuit lawyer = new Lawyer(xiaomin);

        DynamicProxy proxy = new DynamicProxy(xiaomin);

        //获取被代理类小民的classLoader
        ClassLoader loader = xiaomin.getClass().getClassLoader();

        //动态构造一个代理者律师
        ILawsuit lawyer = (ILawsuit) Proxy.newProxyInstance(loader,new Class[]{ILawsuit.class},proxy);

        //律师提交诉讼申请
        lawyer.submit();
        //律师进行举证
        lawyer.burden();
        //律师代替小民进行辩护
        lawyer.defend();
        //完成诉讼
        lawyer.finish();

    }
}


我们在Android项目中使用第三方库Retrofit,使用就是代理模式

ApiService service = retrofit.create(ApiService.class); //创建一个代理实例
我们点进去create方法看看

@SuppressWarnings("unchecked") // Single-interface proxy creation guarded by parameter safety.
  public <T> T create(final Class<T> service) {
    Utils.validateServiceInterface(service);
    if (validateEagerly) {
      eagerlyValidateMethods(service);
    }
    return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[] { service },
        new InvocationHandler() {
          private final Platform platform = Platform.get();

          @Override public Object invoke(Object proxy, Method method, @Nullable Object[] args)
              throws Throwable {
            // If the method is a method from Object then defer to normal invocation.
            if (method.getDeclaringClass() == Object.class) {
              return method.invoke(this, args);
            }
            if (platform.isDefaultMethod(method)) {
              return platform.invokeDefaultMethod(method, service, proxy, args);
            }
            ServiceMethod<Object, Object> serviceMethod =
                (ServiceMethod<Object, Object>) loadServiceMethod(method);
            OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args);
            return serviceMethod.callAdapter.adapt(okHttpCall);
          }
        });
  

可以看到,也是通过Proxy.newProxyInstance生成了一个代理类。

相关文章

  • 设计模式

    单例模式 模板方法模式 工厂模式 代理模式 静态代理 JDK动态代理

  • 设计模式

    单例模式 代理模式 静态代理 jdk动态代理 cglib动态代理 工厂模式 适配器模式 建造者模式 观察者模式

  • kube-proxy的3种模式

    userspace代理模式 iptables代理模式 IPVS代理模式 https://kubernetes.io...

  • 第4章 结构型模式-代理模式

    一、代理模式简介 二、代理模式3个角色 三、代理模式的优点 四、代理模式的实例(游戏代练)

  • 理解代理模式

    原创博客地址 简介 代理模式,也叫做委托模式,分为:静态代理动态代理 代理模式也是平时比较常用的设计模式之一,代理...

  • 结构型 代理模式(文末有项目连接)

    1:什么是代理模式 2:没用代理模式时的实例 3:使用代理模式将其解耦(静态代理) 3:使用代理模式将其解耦(动态...

  • 设计模式-动态代理模式

    之前介绍了代理模式,大家也都了解了代理模式,不过之前介绍的代理模式是静态代理,静态代理什么意思?静态代理指的是代理...

  • 代理模式

    一、什么是代理模式 代理模式(Proxy pattern):代理模式又叫委托模式,是为某个对象提供一个代理对象,并...

  • 设计模式之代理模式(Proxy模式)

    代理模式的引入 代理模式的实例程序 代理模式的分析 代理模式的引入 Proxy是代理人的意思,指的是代替别人进行工...

  • Java设计模式之代理模式

    Java设计模式之代理模式 代理模式 静态代理 动态代理 为什么需要代理 通过代理,我们能够不用知道委托人是谁,而...

网友评论

      本文标题:代理模式

      本文链接:https://www.haomeiwen.com/subject/vjoaactx.html