当前位置: 首页 > news >正文

设计模式篇---代理模式

文章目录

    • 概念
    • 结构
    • 实例
      • 静态代理
      • 动态代理
    • 总结

概念

代理模式:给某一个对象提供一个代理或占位符,并由代理对象来控制对原对象的访问。
比如我们想从其他国家买东西,但我们无法直接联系外国的商家,可以找代理商,让他们帮我们处理,我们是客户端,只需要面向代理商即可,只需要把钱交给代理商,剩下的那些操作,比如联系商家、和商家签订协议等我们都不需要关心。

结构

在这里插入图片描述
Subject(抽象主题):它是代理类和真实类的共同接口,这样一来在任何使用真实对象的地方都可以使用代理对象,客户端通常需要针对抽象主题角色进行编程。
Proxy(代理类) :它包含了真实对象的引用,所以可以在任何时候操作真实对象。一般在调用真实对象前后还需要执行其他操作。
RealSubject(真实类–被代理类):真实类中实现了主要的业务操作。客户端可以调用代理类,来间接的调用真实类。

实例

静态代理

我们想从海外买台电脑,用代理模式实现这个流程。
在这里插入图片描述

购买东西的接口

public interface IBuySomething {void pay();
}

真实类,也就是被代理类

public class Person implements IBuySomething{@Overridepublic void pay() {System.out.println("付款");}
}

代理商,也就是代理类,他来帮我们联系商家

public class Agent implements IBuySomething {private IBuySomething person;public Agent(IBuySomething person) {this.person = person;}@Overridepublic void pay() {findBusiness();person.pay();}private void findBusiness() {System.out.println("我是代理商,付款之前先找到商家");}
}

客户端

public class Client {public static void main(String[] args) {IBuySomething person;person = new Agent(new Person());person.pay();}
}

打印结果:
在这里插入图片描述
也可以让代理商来替我们的朋友来买东西,只需要再声明一个朋友类,让代理商来代理即可。

朋友类

public class Friend implements IBuySomething{@Overridepublic void pay() {System.out.println("我是朋友,我付款");}
}

客户端

public class Client {public static void main(String[] args) {IBuySomething person;person = new Agent(new Friend());person.pay();}
}

打印结果
在这里插入图片描述
反过来,如果想换一家代理商,那就再创建一个新的代理商类即可。

public class Agent2 implements IBuySomething{private IBuySomething person;public Agent2(IBuySomething person) {this.person = person;}@Overridepublic void pay() {findBusiness();person.pay();}private void findBusiness() {System.out.println("我是另外一个代理商,付款之前先找到商家");}
}

动态代理

以上的这种代理方式叫做静态代理。
静态代理的特点是,一个代理类只能代理一个真实类,或者只能代理一个方法。因为它在执行前就编译成了class文件,不会进行改变了,所以被称为静态代理。
但如果我们想动态的代理不同的真实类,或者代理不同的方法,可以使用动态代理来实现。动态代理可以让系统在运行时根据实际需求来动态的创建代理类。
有关动态代理,有两个重要的类。
Proxy类
Proxy类提供了用于创建动态代理对象的方法。它的主要方法newProxyInstance

    public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)

该方法即用来创建一个动态代理对象;第一个参数是代理类的类加载器(作用是将.class文件加载到jvm中,进而生成一个对象实例);第二个参数是真实类实现的接口列表;第三个是执行代理方法的具体程序—InvocationHandler。
简单总结下,要想生成一个代理对象,首先得创建class对象(第一个参数的作用),其次得知道代理谁(第二个参数的作用),最后代理的方法是什么(第三个参数)。这样看来,第二个和第三个参数都是动态的,可变的,这也就是代理模式的灵活性。

InvocationHandler
上面的第三个参数InvocationHandler 是一个接口,它只有一个invoke方法

public Object invoke(Object proxy, Method method, Object[] args)throws Throwable;

该方法用来处理代理类实例的代理方法,并返回相应的结果。即我们代理的方法写在这个方法里。
第一个参数是代理对象;第二个参数是需要代理的方法;第三个参数是需要执行代理方法的参数。

还是拿买东西的例子来说,我们如果买完东西后,发现不合适,需要退款,这时候我们面向的还是代理商,但如果用静态代理的话,那还是需要再写一个代理退款的类,如果业务方法更多的话,那我们需要创建更多的静态代理类,这样处理起来很麻烦。如果用动态代理的话,就没有这么冗余。我们首先实现一个InvocationHandler,它的invoke方法是用来实现代理对象的方法。

public class AgentHandler implements java.lang.reflect.InvocationHandler {private Object object;public AgentHandler(Object object) {this.object = object;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {findBusiness();Object result = method.invoke(object, args);return result;}private void findBusiness() {System.out.println("我是代理商,我要先找到商家");}
}

我们也增加一个退款的方法

public interface IBuySomething {void pay();void refund();
}

客户端调用

public class Client {public static void main(String[] args) {IBuySomething person = new Person();InvocationHandler handler = new AgentHandler(person);Object o = Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(), handler);IBuySomething proxy = (IBuySomething) o;proxy.refund();}
}

打印结果:
在这里插入图片描述

总结

静态代理比较好理解,代理类里面实现了代理的方法。
而动态代理的代理对象是通过Proxy创建的,代理的方法是在InvocationHandler里的invoke方法里,和静态代理对比的话,代理类和代理方法是分离开的。

相关文章:

  • Kafka 最佳实践:构建可靠、高性能的分布式消息系统
  • 基于JAVA+SpringBoot+Vue的前后端分离的医院信息智能化HIS系统
  • linux下查看文件当下的所有文件的大小和查找大文件
  • 【Spring Boot 源码学习】ApplicationListener 详解
  • JWT的原理
  • pyqt5使用Designer实现按钮上传图片
  • Vue 纯css方式实现自定义进度条组件
  • SQL注入概述
  • ffmpeg6.0之ffprobe.c源码分析二-核心功能源码分析
  • git 常用的使用方法
  • 初识Redis
  • 「Verilog学习笔记」多bit MUX同步器
  • Java 中的抽象类与接口:深入理解与应用
  • 配置端口安全示例
  • 探索无监督域自适应,释放语言模型的力量:基于检索增强的情境学习实现知识迁移...
  • 【跃迁之路】【519天】程序员高效学习方法论探索系列(实验阶段276-2018.07.09)...
  • 2017-08-04 前端日报
  • Dubbo 整合 Pinpoint 做分布式服务请求跟踪
  • electron原来这么简单----打包你的react、VUE桌面应用程序
  • exif信息对照
  • httpie使用详解
  • HTTP中的ETag在移动客户端的应用
  • Leetcode 27 Remove Element
  • LeetCode算法系列_0891_子序列宽度之和
  • Linux链接文件
  • Xmanager 远程桌面 CentOS 7
  • Zsh 开发指南(第十四篇 文件读写)
  • 互联网大裁员:Java程序员失工作,焉知不能进ali?
  • 前端代码风格自动化系列(二)之Commitlint
  • 【运维趟坑回忆录 开篇】初入初创, 一脸懵
  • Mac 上flink的安装与启动
  • 翻译 | The Principles of OOD 面向对象设计原则
  • #pragma预处理命令
  • #Ubuntu(修改root信息)
  • #vue3 实现前端下载excel文件模板功能
  • #图像处理
  • $$$$GB2312-80区位编码表$$$$
  • (1)SpringCloud 整合Python
  • (4)事件处理——(2)在页面加载的时候执行任务(Performing tasks on page load)...
  • (echarts)echarts使用时重新加载数据之前的数据存留在图上的问题
  • (二)十分简易快速 自己训练样本 opencv级联lbp分类器 车牌识别
  • (二)延时任务篇——通过redis的key监听,实现延迟任务实战
  • (二十三)Flask之高频面试点
  • (四)JPA - JQPL 实现增删改查
  • (原)本想说脏话,奈何已放下
  • (转)清华学霸演讲稿:永远不要说你已经尽力了
  • (转)我也是一只IT小小鸟
  • ***测试-HTTP方法
  • ./configure,make,make install的作用(转)
  • ./和../以及/和~之间的区别
  • .NET “底层”异步编程模式——异步编程模型(Asynchronous Programming Model,APM)...
  • .NET Core 中插件式开发实现
  • .net wcf memory gates checking failed
  • .NET/C# 避免调试器不小心提前计算本应延迟计算的值
  • .Net多线程Threading相关详解