设计模式之代理模式(1)

news2024/9/29 21:34:14

目录

  • 概述
    • 定义
    • 应用场景
    • 主要角色
    • 类图
  • 详述
    • 基本代码
    • 应用实例
    • 符合的设计原则
  • 总结

概述

定义

    代理模式是一种结构型设计模式,它允许通过一个代理对象来控制对原始对象的访问。代理对象可以在不改变原始对象的情况下,增加一些额外的功能,例如权限验证、缓存等。

应用场景

代理模式常用于以下几种情况:

远程代理:代理对象控制对远程对象的访问,例如远程服务调用。
虚拟代理:代理对象代表了一些昂贵或资源消耗大的对象,延迟加载原始对象。
安全代理:代理对象控制对原始对象的访问权限,例如权限验证。

主要角色

  • 目标接口(Subject Interface):定义了目标对象和代理对象共同实现的接口或抽象类。目标接口规定了客户端可以通过代理对象访问的方法。

  • 目标对象(Real Subject):实际执行业务逻辑的对象,是代理对象所代表的真正对象。目标对象实现了目标接口,代理对象将会委托目标对象执行具体的操作。

  • 代理对象(Proxy):代理对象实现了目标接口,并持有一个对目标对象的引用。代理对象在客户端和目标对象之间起到中介的作用,它可以在调用目标对象之前或之后添加额外的逻辑,以实现对目标对象的控制和管理。

  • 客户端(Client):使用代理对象的对象。客户端通过代理对象来访问目标对象的方法,而无需直接与目标对象交互

    在代理模式中,客户端通过代理对象与目标对象进行交互,代理对象在必要时会进行额外的处理。代理对象可以隐藏目标对象的具体实现细节,提供额外的功能或限制访问权限,从而实现对目标对象的保护和控制。

类图

在这里插入图片描述

详述

基本代码

被代理对象


public class RealSubject implements Subject {
    @Override
    public void doSomething() {
        System.out.println("RealSubject is doing something.");
    }
}

    创建一个接口,作为代理对象和目标对象共同实现的接口

    创建接口的目的是定义代理对象和目标对象共同实现的契约或协议。这个接口定义了代理对象和目标对象之间的通信规范,规定了代理对象需要实现的方法。

    通过定义一个接口,可以将代理对象和目标对象解耦,使得它们可以独立开发和演化。代理对象和目标对象都实现了相同的接口,这意味着它们具有相同的方法签名和行为,可以互相替换使用。

public interface Subject {
    void doSomething();
}

    创建一个代理对象类,实现目标接口,并持有一个对目标对象的引用

    代理对象充当了一个中间人的角色,在客户端和真正执行任务的目标对象之间进行通信和协调。

    代理对象并不是真正执行任务的人,它只是负责管理和控制对目标对象的访问。代理对象可以在执行任务前后添加额外的逻辑或功能,例如权限验证、缓存、日志记录等。

    被代理对象才是真正执行任务的人,它实现了具体的业务逻辑。代理对象在接收到客户端的请求后,会将任务委派给目标对象(被代理对象)来执行。这样可以将任务的执行与具体的业务逻辑分离开来,使得代理对象可以提供一些额外的服务或控制,同时保持目标对象的独立性和可复用性。

public class ProxySubject implements Subject {
    private RealSubject realSubject;

    @Override
    public void doSomething() {
        if (realSubject == null) {
            realSubject = new RealSubject();
        }
        // 在这里可以继续对 realSubject 进行操作
        realSubject.doSomething();
    }

    // 其他代码...
}

注意:代理类当中,为什么要有一个判空的代码?

    第一、这段代码示例中的判空操作称为“延迟初始化”(Lazy Initialization)。延迟初始化是一种性能优化策略,它推迟了对象的创建直到真正需要该对象时才进行。在代理模式的上下文中,这种方式特别有用,因为它允许系统延迟创建计算成本高或者资源消耗大的对象。比如

  • 节约资源:如果realSubject对象的创建成本很高(例如,需要大量内存或时间),那么只有在实际需要使用realSubject对象时才创建它,可以避免在realSubject尚未被使用时就占用宝贵的系统资源。

  • 提高性能:如果realSubject对象在程序运行期间可能根本不会被用到,那么使用延迟初始化可以提高程序启动速度和运行效率,因为避免了不必要的初始化开销。

    第二、代理类通常负责管理实际对象的生命周期,包括实际对象的创建。判空操作就是代理类确保只在首次需要时创建实际对象的一种方式。这样做的好处是,代理类可以在不影响客户端使用的前提下,控制实际对象的初始化过程。

    第三、在实际应用当中是不应该有判空的,因为实际应用当中是被代理类已经存在的,是应该通过依赖倒置注入进来。
再次,在这个里面,判空除了可能想使用原有的被代理类,还可能防止冲突的发生,比如代理,除了代理方法,还有可能代理属性,那么原有的被代理类当中的属性更改之后,如果不判空再创建一个新的被代理类的对象,就会发生冲突。这个的前提是在一个大类当中,这个被代理类没有被回收掉。


    客户端通过代理对象来请求执行任务,并且代理对象会在必要时将请求传递给目标对象。

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

应用实例

    业务场景:一个追求者(Pursuit)通过代理(Proxy)向心仪的女孩(SchoolGirl)送礼物。

    IGiveGift 接口:定义了送礼物的行为,包括送洋娃娃(giveDolls)、送鲜花(giveFlowers)和送巧克力(giveChocolate)。

public interface IGiveGift {
    void giveDolls();
    void giveFlowers();
    void giveChocolate();
}

    Pursuit 类:追求者类,实现了 IGiveGift 接口,具体执行送礼物的动作。构造函数需要传入一个 SchoolGirl 对象,表示追求者要送礼物的对象。

public class Pursuit implements IGiveGift {
    private SchoolGirl mm;
    public Pursuit(SchoolGirl mm){
        this.mm=mm;
    }
    public void giveDolls(){
        System.out.println(this.mm.getName()+",你好,送你洋娃娃");
    }
    public void giveFlowers(){
        System.out.println(this.mm.getName()+",你好,送你鲜花");
    }
    public void giveChocolate(){
        System.out.println(this.mm.getName()+",你好,送你巧克力");
    }
}

    Proxy 类:代理类,持有追求者(Pursuit)的引用,并且对外提供与 IGiveGift 接口相同的方法。当调用代理的送礼物方法时,实际上是调用追求者的对应方法。 它的构造函数接收一个 SchoolGirl 对象,并创建一个 Pursuit 对象来初始化追求者。

public class Proxy {
    private Pursuit gg;
    private SchoolGirl mm;
    public Proxy(SchoolGirl mm){//代理认识被追求者
       this.gg=new Pursuit(mm);//代理初始化过程中,实际是追求者初始化的过程

    }
    public void giveDolls(){
        gg.giveDolls();
    }
    public void giveFlowers(){
       gg.giveFlowers();
    }
    public void giveChocolate(){
       gg.giveChocolate();
    }
}

    SchoolGirl 类:被追求的女孩类,拥有名字属性和相应的获取及设置方法。

public class SchoolGirl {
    private String name;
    public String getName(){
        return this.name;
    }
    public void setName(String name){
        this.name=name;
    }
}

    Client ,首先创建了一个 SchoolGirl 对象 girlLili,并设置了名字为“丽丽”。然后创建了一个 Proxy 对象 boyDL,并通过代理对象调用送礼物的方法。客户端不需要知道实际对象(Pursuit 类)的实现细节,只需要与代理对象交互。接触耦合。

public class Client {
    public static void main(String[] args) {
    SchoolGirl girlLili=new SchoolGirl();
        girlLili.setName("丽丽");

        Proxy boyDL=new Proxy(girlLili);
        boyDL.giveDolls();
        boyDL.giveChocolate();
        boyDL.giveFlowers();
    }
}

符合的设计原则

  • 单一职责原则(Single Responsibility Principle):一个类应该只有一个引起变化的原因。在代理模式中,代理类(Proxy)负责控制对实际对象的访问,而实际对象(如Pursuit类)则专注于执行其核心业务逻辑。

  • 开闭原则(Open/Closed Principle):软件实体应当对扩展开放,对修改关闭。代理模式允许在不修改实际对象代码的情况下,通过代理类来扩展功能。例如,可以添加新的代理类来实现不同的访问控制策略。

  • 接口隔离原则(Interface Segregation Principle):客户端不应该依赖它不需要的接口。在代理模式中,代理类和实际对象都实现相同的接口(IGiveGift),客户端仅与该接口进行交互,而不是直接与实现细节打交道。

  • 依赖倒转原则(Dependency Inversion Principle):高层模块不应该依赖低层模块,两者都应该依赖于抽象;抽象不应该依赖于细节,细节应该依赖于抽象。在代理模式中,客户端代码(Client)依赖于接口(IGiveGift),而不是具体的类(Pursuit或Proxy),这样就可以灵活地替换或修改具体的实现而不影响客户端。

  • 合成复用原则(Composite Reuse Principle):尽量使用对象组合,而不是继承来达到复用的目的。代理模式中,代理类通过包含一个实际对象的引用来实现功能,而不是通过继承实际对象来扩展功能。

  • 最少知识原则(Least Knowledge Principle)或迪米特法则(Law of Demeter):一个对象应该对其他对象有尽可能少的了解。在代理模式中,客户端不需要知道实际对象如何实现或者如何被访问的细节,它只需要与代理对象交互,从而减少了系统中各部分之间的耦合。

总结

    代理模式是一种常用的设计模式,它通过代理对象在保护和控制原始对象访问上起到中间层的作用。今天只讲了静态代理,也就是在编译时就确定了代理对象和原始对象的关系,下次会接着讲动态代理,可以在运行时动态生成代理对象,还有JDK动态代理和CGLIB动态代理的区别。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1283644.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

kubectl获取ConfigMap导出YAML时如何忽略某些字段

前言: 当我们在使用Kubernetes时,常常需要通过kubectl命令行工具来管理资源。有时我们也想将某个资源的配置导出为YAML文件,这样做有助于版本控制和资源的迁移。然而,默认情况下,使用kubectl get命令导出资源配置会包…

【IPv6】IPv6协议

一、IPv6数据报格式 这是与v4报头的对比 1.8bit的版本保留了,v4版本就是4,v6就是6。 2.v6去除了v4的首部长度字段,因为v6的首部长是固定的40字节。 3.服务类型(Type of Service, ToS)和通信类型(Traffi…

【Linux下如何生成coredump文件】

一,什么是coredump 我们经常听到大家说到程序core掉了,需要定位解决,这里说的大部分是指对应程序由于各种异常或者bug导致在运行过程中异常退出或者中止,并且在满足一定条件下(这里为什么说需要满足一定的条件呢&#…

贝叶斯网络 (人工智能期末复习)

文章目录 贝叶斯网络(概率图模型)定义主要考点例题- 要求画出贝叶斯网络图- 计算各节点的条件概率表- 计算概率- 分析独立性 贝叶斯网络(概率图模型) 定义 一种简单的用于表示变量之间条件独立性的有向无环图(DAG&am…

操作系统概述及发展史、Linux内核、发行版及应用领域

一、 操作系统(Operation System,OS) 裸机:没有安装操作系统的计算机 如果想在 裸机 上运行自己所编写的程序,就必须用机器语言书写程序如果计算机上安装了操作系统,就可以在操作系统上安装支持的高级语言…

工程师业余生活之制作蔬菜盆景

工程师业余生活陶冶情操之制作蔬菜盆景 (蔬 果 盆 景 裝 點 家 居) 市場上好多蔬菜瓜果,稍用一些心思,將一些價廉的蔬果製成別致的盆景, 便能使家居充滿自然氣息,增添生活情趣。以下介紹幾種製作方法: 【番薯盆景】 (番薯又名地…

人工智能_机器学习060_核函数对应数学公式_数据空间错位分割_简单介绍_以及核函数总结---人工智能工作笔记0100

我们之前做的都是线性分类问题,那么需要一根线来分割类别,但是,如果出现了,环形数据,我们知道,在二维中我们就无法分割了,那么有没有什么办法分割呢? 实际上是有的,可以看到,我们可以把数据进行升维,可以看到,如果把数据升高到2维度以上,可以看到,神奇的一幕出现了,这个时候,因…

每日3道PWN(第一天)

环境准备 我现在用的是kali 现阶段工具:checkesc、IDA、比较完善的python环境 下载工具的话,我这里不提供了 buuctf——test_your_nc1 参考wp: BUUCTF PWN-----第1题:test_your_nc_buuctf test_your_nc-CSDN博客 查看的资料:…

风格迁移网络修改流程(自用版)

一. AdaAttN-Revisit Attention Mechanism in Arbitrary Neural Style Transfer(ICCV2021) 下载vgg_normalised.pth打开visdom python -m visdom.server在 train_adaattn.sh 中配置 content_path、style_path 和 image_encoder_path,分别表…

深入理解Zookeeper系列-4.Watcher原理

👏作者简介:大家好,我是爱吃芝士的土豆倪,24届校招生Java选手,很高兴认识大家📕系列专栏:Spring源码、JUC源码、Kafka原理、分布式技术原理🔥如果感觉博主的文章还不错的话&#xff…

深入浅出理解kafka

1.Kafka简介 Kafka 本质上是一个 MQ(Message Queue),使用消息队列的优点: 解耦:允许独立的扩展或修改队列两边的处理过程。可恢复性:即使一个处理消息的进程挂掉,加入队列中的消息仍然可以在系…

LinuxBasicsForHackers笔记 -- 控制文件和目录权限

对于每个文件和目录,我们可以指定文件所有者、特定用户组以及所有其他用户的权限状态。 不同类型的用户 在Linux中,root用户是拥有一切权力的。 root 用户基本上可以在系统上执行任何操作。 系统上的其他用户具有有限的能力和权限,并且几乎…

使用正则表达式时-可能会导致性能下降的情况

目录 前言 正则表达式引擎 NFA自动机的回溯 解决方案 前言 正则表达式是一个用正则符号写出的公式,程序对这个公式进行语法分析,建立一个语法分析树,再根据这个分析树结合正则表达式的引擎生成执行程序(这个执行程序我们把它称作状态机&a…

GitHub项目推荐-Deoldify

有小伙伴推荐了一个老照片上色的GitHub项目,看了简介,还不错,推荐给大家。 项目地址 GitHub - SpenserCai/sd-webui-deoldify: DeOldify for Stable Diffusion WebUI:This is an extension for StableDiffusions AUTOMATIC1111 w…

IDEA 下载mysql驱动下载在不下来

结合一下 https://www.cnblogs.com/dadian/p/11936056.htmlhttps://www.cnblogs.com/dadian/p/11936056.html并且下载的 在idea改名 加入 加入到库 等待一会就要你输入sql的root和密码了,就OK

React创建项目

React创建项目 提前安装好nodejs再进行下面的操作,通过node -v验证是否安装 1.设置源地址 npm config set registry https://registry.npmmirror.com/2.确认源地址 npm config get registry返回如下 https://registry.npmmirror.com/3.输入命令 npx create-re…

SpringCloud | Dubbo 微服务实战——注册中心详解

前言 「作者主页」:雪碧有白泡泡 「个人网站」:雪碧的个人网站 |Eureka,Nacos,Consul,Zookeeper在Spring Cloud和Dubbo中实战 引言 在项目开发过程中,随着项目不断扩大,也就是业务的不断增多,我们将采用集群&#xf…

(C语言)通过循环按行顺序为一个矩阵赋予1,3,5,7,9,等奇数,然后输出矩阵左下角的值。

#include<stdio.h> int main() {int a[5][5];int n 1;for(int i 0;i < 5;i ){for(int j 0;j < 5;j ){a[i][j] n;n 2;}}for(int i 0;i < 5;i ){for(int j 0;j < i;j )printf("%-5d",a[i][j]);printf("\n");}return 0; } 运行截图…

Win7 旗舰版打开Rustdesk软件提示无法启动程序 ,计算机中丢失api-ms-win-shcore-scaling-|1-1-1.dll

环境: Win7 旗舰版 64位 Rustdesk1.19自编译客户端 问题描述: Win7 旗舰版打开Rustdesk软件提示无法启动程序 ,计算机中丢失api-ms-win-shcore-scaling-|1-1-1.dll "api-ms-win-shcore-scaling-|1-1-1.dll" 是一个系统动态链接库文件,它是Windows操作系统的一…