GPT带我学-设计模式-代理模式

2023-11-11

什么是代理模式

代理模式(Proxy Pattern)是设计模式中的一种结构型模式,它为其他对象提供一种代理以控制对这个对象的访问。

代理模式有三个主要角色:抽象主题(Subject)、真实主题(Real Subject)和代理(Proxy)。

抽象主题(Subject):定义了真实主题与代理之间的共同接口,这样真实主题和代理可以互相替换使用。

真实主题(Real Subject):指实际要被代理的对象。

代理(Proxy):提供了一个与真实主题相同的接口,它内部持有一个真实主题的引用,并在真实主题的实例化或执行某些操作前后进行控制或扩展。

代理模式的主要目的是在不改变真实主题的情况下,为其提供一个代理类来进行访问控制或增强功能。代理模式可以实现懒加载,即在真正需要使用真实主题的时候才进行实例化,从而节省资源和提高性能。代理模式还能提供额外的功能,例如在调用真实主题的方法前后进行日志记录、安全控制、事务管理等。

代理模式可以分为静态代理和动态代理。静态代理是通过手动编写代理类来实现的,而动态代理则是在运行时动态生成代理类。动态代理通常使用Java的反射机制来实现,可以减少冗余代码,并且对真实主题的访问完全透明。

总之,代理模式提供了一种间接访问对象的方式,可以控制对象的访问、增强对象的功能,并且可以在不改变真实对象的情况下进行扩展。

代理模式有几种形式

在设计模式中,代理模式有两种常见的形式:

静态代理(Static Proxy):静态代理需要手动编写代理类,代理类和真实主题类实现相同的接口或继承相同的父类,并在代理类中持有一个真实主题类的引用。静态代理在编译时就确定了代理对象和真实对象的关系。

动态代理(Dynamic Proxy):动态代理在运行时动态生成代理类,不需要手动编写代理类。Java中的动态代理通常使用Java反射机制实现,可以通过Proxy类和InvocationHandler接口来创建动态代理对象。动态代理可以代理接口或类,适用于需要代理多个不同类型对象的场景。

在静态代理和动态代理中,代理类都是通过调用真实主题类的方法,并在方法前后加上一些额外的逻辑(如日志记录、权限验证等)来控制或增强真实主题的功能。

除了静态代理和动态代理,还有一些其他的代理变体形式,如远程代理、虚拟代理、保护代理等,这些形式根据具体的应用场景和需求进行使用。

请举一个具体的静态代理例子

假设有一个接口 Subject,定义了一个抽象主题的操作方法 request()。现在我们需要用静态代理来实现在真实主题的执行前后进行日志记录。

首先,定义一个 Subject 接口:

public interface Subject {
    void request();
}

接下来,创建真实主题类 RealSubject,实现 Subject 接口:

public class RealSubject implements Subject {
    @Override
    public void request() {
        System.out.println("真实主题执行请求");
    }
}

然后,创建一个代理类 ProxySubject,实现 Subject 接口,并在 request() 方法的前后添加日志记录的逻辑:

public class ProxySubject implements Subject {
    private RealSubject realSubject;

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

    @Override
    public void request() {
        System.out.println("代理主题执行请求之前的日志记录");
        
        // 调用真实主题的方法
        realSubject.request();
        
        System.out.println("代理主题执行请求之后的日志记录");
    }
}

最后,我们可以通过以下代码来使用静态代理:

public class Main {
    public static void main(String[] args) {
        RealSubject realSubject = new RealSubject();
        ProxySubject proxy = new ProxySubject(realSubject);
        proxy.request();
    }
}

运行程序,输出结果如下:

代理主题执行请求之前的日志记录
真实主题执行请求
代理主题执行请求之后的日志记录

可以看到,通过静态代理模式,我们在真实主题的执行前后添加了日志记录的逻辑,实现了对真实主题的控制和功能增强,而不需要修改真实主题类的代码。这是静态代理模式的一种常见应用场景。

同样的例子怎么改成动态代理模式呢

要将上述示例改为动态代理模式,需使用Java的动态代理实现。首先,需要借助 java.lang.reflect.Proxy 类和 java.lang.reflect.InvocationHandler 接口。

首先,定义 Subject 接口和 RealSubject 真实主题类,与上述示例相同。

接下来,创建一个实现 InvocationHandler 接口的代理处理器类 ProxyHandler,用于在真实主题的方法执行前后添加日志记录的逻辑。在 invoke() 方法中,可以定义增强的逻辑以及调用真实主题的方法。

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class ProxyHandler implements InvocationHandler {
    private Object realSubject;

    public ProxyHandler(Object realSubject) {
        this.realSubject = realSubject;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("代理主题执行请求之前的日志记录");

        // 调用真实主题的方法
        Object result = method.invoke(realSubject, args);

        System.out.println("代理主题执行请求之后的日志记录");

        return result;
    }
}

最后,可以通过以下代码来使用动态代理:

import java.lang.reflect.Proxy;

public class Main {
    public static void main(String[] args) {
        RealSubject realSubject = new RealSubject();
        InvocationHandler handler = new ProxyHandler(realSubject);
        
        // 创建动态代理对象
        Subject proxySubject = (Subject) Proxy.newProxyInstance(
            realSubject.getClass().getClassLoader(),
            realSubject.getClass().getInterfaces(),
            handler
        );
        
        proxySubject.request();
    }
}

运行程序,输出结果与静态代理的例子相同:

代理主题执行请求之前的日志记录
真实主题执行请求
代理主题执行请求之后的日志记录

可以看到,通过动态代理模式,我们实现了在真实主题的执行前后添加日志记录的逻辑,而不需要直接操作代理类,使代理对象动态生成,并且能够透明地调用真实主题的方法。这是动态代理模式的一种常见应用场景。

补充说明(《Head First设计模式》)

java在java.lang.reflect包中有自己的代理实现方式,通过相关api可以动态创建一个代理类。
通过代理类去调用被代理的方法。
因为实际代理类是在运行时创建的,所以把java的这个技术叫做动态代理。
Java为我们动态创建了Proxy。我们通过实现InvocationHandler接口告诉Proxy去做什么。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

GPT带我学-设计模式-代理模式 的相关文章

  • C语言-数字爆炸游戏

    问题 你好 欢迎来到数字爆炸 系统随机生成一个数字 猜大了 提示猜大了 猜小了 提示猜小了 思路 先写游戏大概思路 首先 会有菜单吧 所以先写一个菜单函数 显示菜单的内容 然后选择菜单中的选项 不同的选项 对应不同的功能 因此这个用swit
  • IAP升级应用程序流程和原理(前提是:用户要先编写的boot loader程序)

    STM32升级方法 一 IAP升级 骑着蜗牛写代码的博客 CSDN博客 stm32 iap升级 IAP升级是使用客户自己编写的boot loader而不是系统自带的 所以flash需要为两部分 一部分用来存放IAP boot loader
  • 通信协议——MII/GMII

    MAC层接口协议 概述 MII GMII 概述 随着宽带数据业务的飞速普及 数据产品的端口速率及端口密度也大大提升 于是以太网的MAC与PHY层之间的媒介无关接口 MII 也在不断简化 推陈出新 多种多样 基于此 本文简要介绍了以太网的PH

随机推荐

  • STM32芯片--FSMC外设扩展外部SRAM

    这里写自定义目录标题 一 为什么要扩展外部SRAM 二 什么是SRAM 简介 存储器型号 容量 原理框图 引脚配置 通讯方式 读写特性 读取数据时序图 读取数据的时序要求 写入数据时序图 写入数据的时序要求 三 什么是FSMC外设 简介 F
  • mac系统更新后无法打开axure8/9

    吐槽一下难用的mac 系统更新后 axure打都打不开了 双击一下 好的时候 报个错 Exception has been thrown by the target of an invocation 不好的时候 错也不报 直接打不开 甚至关
  • 密码学——1.密码学概论

    1 基本术语 1 1 密码使用学 ceyptography vs 密码编码学 crypyology 密码使用学是一种为了达到隐藏消息含义而使用秘文写的一门科学 密码分析学是一门研究在不知道通常解密所需要的秘密信息的情况下对加密的信息进行解
  • nacos 集群部署

    nacos 集群部署 之前 nacos 都是单机部署 使用倒还稳定 没有因为 nacos 出过问题 最近因为各种原因连续迁了几次服务器 中间涉及 nacos 服务器 比较麻烦 所以决定把 nacos 改为集群部署 nacos官方也推荐生产环
  • 【面试题】Javascript的这些运算符,你都都掌握哪些?

    前端面试题库 面试必备 推荐 地址 前端面试题库 国庆头像 国庆爱国 程序员头像 总有一款适合你 theme devui blue highlight a11y light 无论是JavaScript还是其他语言 运算符是基础 表达式和语句
  • python安装easygui

    1 下载地址 https sourceforge net projects easygui files 0 96 2 双击easygui文件夹 打开后复制或剪切easygui py文件放到你原来安装python的文件夹里 python Li
  • DNS解析分类

    DNS 域名系统 解析是将域名转换为对应的IP地址的过程 根据不同的功能和角色 DNS解析可以分为以下几种分类 递归解析 Recursive Resolution 递归解析是指DNS客户端向本地DNS服务器 如ISP提供的DNS服务器 发送
  • 图像配准之特征点匹配的思考

    最近赶时髦 看了一些智能优化算法如蚁群算法 还有机器学习的一些东西 就想着怎么把这些先进的东西用在图像配准中 头脑风暴了一下 觉得在已经检测到两幅图像的特征点的基础上 就如何对它们进行匹配似乎有优化的空间 匹配的过程其实是找对应点的过程 对
  • 操作系统考试重点

    1 什么是进程 什么是线程 进程与线程有何区别 进程 进程是具有独立功能的程序在数据集合上的一次执行过程 线程 线程是进程内的一个执行实体或执行单元 区别 1 进程的地址空间是独立的 而同一进程内的线程共享同一地址空间 一个进程内的线程在另
  • STL源码剖析——deque的实现原理和使用方法详解

    Deque 简介 deque是 double ended queue 的缩写 和vector一样都是STL的容器 deque 是双端数组 而 vector 是单端的 deque 在接口上和 vector 非常相似 在许多操作的地方可以直接替
  • 【IOS】移动端设置input只能输入数字,在IOS系统没有效果

    在移动端设置input的type为number类型的时候需要区分一下安卓还是IOS 如果只设置type为number类型的时候 在安卓生效 但是IOS还是会有可以输入汉字的问题出现 所以就需要一个新的属性 pattern 0 9 来控制输入
  • BigDecimal的使用小结

    文章目录 1 为什么用 BigDecimal 2 构造函数的选择 3 加减乘除的使用 4 保留小数 5 RoundingMode类 6 其他方法 1 为什么用 BigDecimal 因为 double float的计算很不靠谱 莫名其妙的会
  • 《微积分基础》学习(一)

    本系列文章主要记录我在学习coursera上的 Calculus One 在线课程的笔记 该课程是俄亥俄州立大学的经典课程 教授的发音比较标准 授课方式轻松愉悦 是不可错过的 微积分 入门课程 本文直接从第二周切入 第一周是课程介绍和学习方
  • intelli idea中配置Tomcat找不到的解决办法

    这两天新入职一家公司 公司用的是intelli idea 以前用习惯了eclipse 感觉到有点不太习惯 当然 intelli idea也有自己的强大之处 在开始配置Tomact之前 按照网上的说法 发现点击 号之后没有Tomcat 于是乎
  • printf 和scanf

    1 printf 简介 1 1 printf 的格式 printf 函数的原型为 include
  • Xbox One 升级后黑屏修复

    好久没用Xbox了 近期突然想要利用体感游戏进行锻炼 结果打开后提示需要更新 更新包有4G左右 随手选了更新后就让xbox后台更新了 过了一段时间切回hdmi信号发现一片黑屏 手柄的xbox键可以唤出关闭菜单 但是其他操作都无效 经过一番研
  • 20230322 元宇宙

    VR AR MR XR的区别和联系 知乎
  • 解决Kaggele无法下载输出output文件夹下的文件

    import os os chdir kaggle working print os getcwd print os listdir kaggle working from IPython display import FileLink F
  • 支付宝给个人账号转账付款

    一 说明 转账到支付宝账户是为了满足支付宝商户向其他支付宝账户进行单笔转账的需求 针对具备开发能力的商户 提供通过 API 接口完成单笔转账的功能 商家只需输入另一个正确的支付宝账号 即可将单笔资金从本人的支付宝账户转账至另一个支付宝账户
  • GPT带我学-设计模式-代理模式

    什么是代理模式 代理模式 Proxy Pattern 是设计模式中的一种结构型模式 它为其他对象提供一种代理以控制对这个对象的访问 代理模式有三个主要角色 抽象主题 Subject 真实主题 Real Subject 和代理 Proxy 抽