代理模式(静态代理/动态代理)

代理模式(Proxy Pattern)

一 定义

为其他对象提供一种代理,以控制对这个对象的访问。
代理对象在客户端和目标对象之间起到了中介作用,起到保护或增强目标对象的作用。
属于结构型设计模式。

代理模式分为静态代理和动态代理。
静态代理是显式声明被代理对象,即硬编码出来的代理结构;
动态代理是动态配置和替换被代理对象,即通过在jvm中生成一个代理类来实现代理。

代理模式标准示例:

在这里插入图片描述
顶层会有 subject的接口,RealSubject 和 Proxy 都实现了 Subject接口。
Proxy中,拥有 RealSubject对象的引用,在Proxy的构造方法中,将RealSubject作为参数传入,然后在Proxy 的同名方法中,调用 RealSubject的方法。但是在调用 RealSubject的方法前后,可以加入Proxy的自有逻辑。

上述类图中各个类的代码如下:
subject 接口类:ISubject

public interface ISubject{
	void request();
}

被代理的类:RealSubject

public class RealSubject implements ISubject{
	public void request(){
		System.out.println("real reqeust");
	}
}

代理类:Proxy

public class Proxy implements ISubject{
	private ISubject target;
	public Proxy(ISubject target){
		this.target = target;
	}
	public void request(){
		before();//调用前逻辑
		target.request();
		after();//调用后逻辑
	}
}

客户端 ProxyClient

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

接下来,我们通过一个实际场景,来呈现出静态代理与动态代理的异同。


阿毛想租房,他首先考虑的是房产中介,因为房源集中在中介那里。

首先是静态代理的实现:

租户接口为:ITenant
中介类为:HouseProxy
实际租户有两个人:AmaoLaoSan
代码如下:

public interface ITenant {
    void require();
}
public class HouseProxy implements ITenant{

    private ITenant custom;

    public HouseProxy(ITenant custom){
        this.custom = custom;
    }
    public void require() {
        before();
        custom.require();
        after();
    }

    private void before() {
        System.out.println("当前代理权限通过验证,可以登录系统开始筛选房源。");
    }

    private void after() {
        System.out.println("按照上述要求找到的房源有:xxx");
        System.out.println();
    }
}
public class Amao implements ITenant {
    public void require() {
        System.out.println("阿毛的要求:一室一厅,2楼,朝南,2000元以内");
    }
}
public class LaoSan implements ITenant{
    public void require() {
        System.out.println("老三的要求:二室一厅,3楼以上,南北通透,4000元以内");
    }
}

客户端调用代码:

public class Client {
    public static void main(String[] args) {
        //中介为阿毛找房:
        HouseProxy proxy = new HouseProxy(new Amao());
        proxy.require();
        //中介为老三找房:
        HouseProxy proxy1 = new HouseProxy(new LaoSan());
        proxy1.require();
    }
}

执行结果为:

当前代理权限通过验证,可以登录系统开始筛选房源。
阿毛的要求:一室一厅,2楼,朝南,2000元以内
按照上述要求找到的房源有:xxx

当前代理权限通过验证,可以登录系统开始筛选房源。
老三的要求:二室一厅,3楼以上,南北通透,4000元以内
按照上述要求找到的房源有:xxx

动态代理的实现

由于租户不是随时可以看房,所以要和中介约好看房时间,所以我们在ITenant 接口中,增加了看房时间。

public interface ITenant {
    void require();
    void lookHouseTime();
}

相应的,Amao和LaoSan 都需要实现这个方法:

public class Amao implements ITenant {
    public void require() {
        System.out.println("阿毛的要求:一室一厅,2楼,朝南,2000元以内");
    }

    public void lookHouseTime() {
        System.out.println("阿毛看房时间:周末");
    }
}
public class LaoSan implements ITenant {
    public void require() {
        System.out.println("老三的要求:二室一厅,3楼以上,南北通透,4000元以内");
    }

    public void lookHouseTime() {
        System.out.println("老三看房时间:周中");
    }
}

最后,是采用JDK动态代理实现的HouseProxy:

public class JDKHouseProxy implements InvocationHandler {

    private ITenant custom;

    public ITenant getInstance(ITenant custom){
        this.custom = custom;
        Class<?> clazz = custom.getClass();
        return (ITenant) Proxy.newProxyInstance(clazz.getClassLoader(),clazz.getInterfaces(),this);
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        if (!method.getName().equals("lookHouseTime")) {
            before();
        }
        Object result = method.invoke(this.custom,args);
        if (!method.getName().equals("lookHouseTime")) {
            after();
        }
        return result;
    }
    private void before() {
        System.out.println("--------------------");
        System.out.println("中介:当前代理权限通过验证,可以登录系统开始筛选房源。");
    }

    private void after() {
        System.out.println("中介:按照上述要求找到的房源有:xxx");
    }

}

这里我们会发现,使用动态代理后,代理中介无需实现ITenant接口,如果ITenant接口发生变化,也不会对代理对象产生影响。
客户端调用类:

public class Client {
    public static void main(String[] args) {
        //中介为阿毛找房:
        JDKHouseProxy proxy = new JDKHouseProxy();
        ITenant tenantAmao = proxy.getInstance(new Amao());
        tenantAmao.require();
        tenantAmao.lookHouseTime();

        //中介为老三找房:
        JDKHouseProxy proxy1 = new JDKHouseProxy();
        ITenant laosan = proxy1.getInstance(new LaoSan());
        laosan.require();
        laosan.lookHouseTime();
    }
}

执行结果:

--------------------
中介:当前代理权限通过验证,可以登录系统开始筛选房源。
阿毛的要求:一室一厅,2楼,朝南,2000元以内
中介:按照上述要求找到的房源有:xxx
阿毛看房时间:周末
--------------------
中介:当前代理权限通过验证,可以登录系统开始筛选房源。
老三的要求:二室一厅,3楼以上,南北通透,4000元以内
中介:按照上述要求找到的房源有:xxx
老三看房时间:周中

动态代理除了JDK的API之外,还有cglib的方式。
二者区别在于,JDK的动态代理需要目标对象有继承体系(即实现接口);而cglib则不需要被代理对象存在继承体系。
以下是cglib的示例:
cglib的中介代理类:CglibHouseProxy

public class CglibHouseProxy implements MethodInterceptor {

    public Object getInstance(Class<?> clazz){
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(clazz);
        enhancer.setCallback(this);
        return enhancer.create();
    }

    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        if (!method.getName().equals("lookHouseTime")) {
            before();
        }

        Object result  = proxy.invokeSuper(obj,args);

        if (!method.getName().equals("lookHouseTime")) {
            after();
        }
        return result;
    }

    private void before() {
        System.out.println("--------------------");
        System.out.println("中介:当前代理权限通过验证,可以登录系统开始筛选房源。");
    }

    private void after() {
        System.out.println("中介:按照上述要求找到的房源有:xxx");
    }
}

客户端调用类:

public class CglibClient {
    public static void main(String[] args) {
        CglibHouseProxy proxy = new CglibHouseProxy();
        ITenant tenant = (ITenant) proxy.getInstance(Amao.class);
        tenant.require();
        tenant.lookHouseTime();

    }
}

执行结果:

--------------------
中介:当前代理权限通过验证,可以登录系统开始筛选房源。
阿毛的要求:一室一厅,2楼,朝南,2000元以内
中介:按照上述要求找到的房源有:xxx
阿毛看房时间:周末

补充:cglib pom.xml的引入

    <dependency>
      <groupId>cglib</groupId>
      <artifactId>cglib-nodep</artifactId>
      <version>2.2</version>
    </dependency>

以上就是本文全部内容。感谢您的阅读。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

视频监控统一管理平台LntonCVS安防视频监控系统视频汇聚方案

LntonCVS平台最初被设计为一个以视频汇聚为核心的平台。那么&#xff0c;什么是视频汇聚平台&#xff0c;以及它是如何处理视频资源的呢&#xff1f;简单来说&#xff0c;视频汇聚平台能够从不同的视频源&#xff08;如直播和点播&#xff09;收集、整合和展示视频内容。以下是…

【2024最新华为OD-C/D卷试题汇总】[支持在线评测] 身高差值排序(100分) - 三语言AC题解(Python/Java/Cpp)

&#x1f36d; 大家好这里是清隆学长 &#xff0c;一枚热爱算法的程序员 ✨ 本系列打算持续跟新华为OD-C/D卷的三语言AC题解 &#x1f4bb; ACM银牌&#x1f948;| 多次AK大厂笔试 &#xff5c; 编程一对一辅导 &#x1f44f; 感谢大家的订阅➕ 和 喜欢&#x1f497; &#x1f…

vue3+element ui +ts 封装周范围选择器

vue3element ui ts 封装周范围选择器 在业务场景中&#xff0c;产品需要在页面中使用周范围选择器&#xff0c;我们在使用ant-design的时候里面是有自带的&#xff0c;但是在emement中只有指定周的范围选择器&#xff1a; 这个是ant-design的周范围选择器 这个是element ui 的…

.net 奇葩问题调试经历之1——在红外相机获取温度时异常

📢欢迎点赞 :👍 收藏 ⭐留言 📝 如有错误敬请指正,赐人玫瑰,手留余香!📢本文作者:由webmote 原创📢作者格言:新的征程,我们面对的不仅仅是技术还有人心,人心不可测,海水不可量,唯有技术,才是深沉黑夜中的一座闪烁的灯塔序言 我们在研发中,经常除了造产品…

C#的Switch语句2(case后的值与模式匹配)

文章目录 switch语法结构case具体的值枚举值字符串const关键字 如果没有匹配的值default语句不一定要在最后 模式匹配与C的差异-case穿透&#xff08;Fall-through&#xff09;下一篇文章 switch语法结构 基础的语法结构&#xff0c;在上一篇文章已经写了&#xff0c;具体请看…

Shiro550 反序列化漏洞(CVE-2016-4437)

目录 Shiro介绍 漏洞原理 判断是否存在漏洞 利用ShiroExploit工具执行命令&#xff1a; 利用shiro-exploit工具综合利用工具执行命令&#xff1a; 这一篇是参考别的师傅的好文章对Shiro550反序列化漏洞的学习和练习 Shiro介绍 Apache Shiro是一个强大易用的java安全框架…

ASP.NET MVC企业级程序设计(增删,页面水平排列,字符串拼接,非空,添加框内默认提示)

目录 题目&#xff1a; 实现过程 控制器代码 DAL BLL Index Deile 题目&#xff1a; 实现过程 控制器代码 using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; using MvcApplication1.Models;namespac…

Tensorflow入门实战 T05-运动鞋识别

目录 一、完整代码 二、训练过程 &#xff08;1&#xff09;打印2行10列的数据。 &#xff08;2&#xff09;查看数据集中的一张图片 &#xff08;3&#xff09;训练过程&#xff08;训练50个epoch&#xff09; &#xff08;4&#xff09;训练结果的精确度 三、遇到的问…

Docker环境离线安装

Docker环境离线安装 下载下列.deb包 sudo *.deb

【PyQt5】python桌面级应用开发:PyQt5介绍,开发环境搭建快速入门

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

天津媒体邀约,及媒体名单?

传媒如春雨&#xff0c;润物细无声&#xff0c;大家好&#xff0c;我是51媒体网胡老师。 媒体宣传加速季&#xff0c;100万补贴享不停&#xff0c;一手媒体资源&#xff0c;全国100城线下落地执行。详情请联系胡老师。 天津作为中国北方的重要城市&#xff0c;拥有丰富的媒体资…

Jenkins+K8s实现持续集成(二)

部署前呢&#xff0c;要先把jenkins搭建好。 同时呢已经有了k8s的环境。 基于以上两步已经有了的情况&#xff0c;继续要实现jenkinsk8s持续集成&#xff0c;需要先准备四个文件&#xff1a; Dockerfile首先要准备好一个Dockerfile文件&#xff0c;用于构建Docker镜像的文本…

最新版本IntelliJ IDEA安装与“坤活”使用

最新版本IntelliJ IDEA安装与“科学”使用 IntelliJ IDEA安装与坤活下载安装坤活idea1.将下面两个压缩文件解压到安装位置&#xff0c;注意路径不要包含中文空格等特殊符号2.双击 install-all-users.vbs &#xff0c;然后点击确定&#xff0c;等到出现 Done的弹窗3. 打开idea复…

函数依赖集等价、最小函数依赖集

一、函数依赖集等价 1、定义 假设F、G为一个关系模式上的两个函数依赖集&#xff0c;若&#xff0c;则称F和G是等价的&#xff0c;也可称F和G 互相覆盖。 2、判断 &#xff08;1&#xff09;引理3&#xff1a; 的充分必要条件是且 &#xff08;2&#xff09;两步走&…

密码学及其应用——为什么选择接近的质数因子对RSA加密算法不安全?

RSA加密算法是一种广泛使用的非对称加密算法&#xff0c;它的安全性依赖于大整数分解的难度。具体来说&#xff0c;RSA算法生成的公钥包含一个大整数N&#xff0c;这是两个大质数p和q的乘积。然而&#xff0c;如果这两个质数p和q太接近&#xff0c;则可以相对容易地对N进行因式…

Study--Oracle-04-SQL练习

一、SQL语句思维导图 二、SQL练习 -- 以employee_id 为排序&#xff0c;列出前5个人 -- FETCH select employee_id,first_name from employees order by employee_id FETCH FIRST 5 rows only; -- 以employee_id 为排序&#xff0c;从第6个人开始 到第10个人 -- offset …

计算机组成原理---Cache的基本工作原理习题

对应知识点&#xff1a; Cache的基本原理 1.某存储系统中&#xff0c;主存容量是Cache容量的4096倍&#xff0c;Cache 被分为 64 个块&#xff0c;当主存地址和Cache地址采用直接映射方式时&#xff0c;地址映射表的大小应为&#xff08;&#xff09;(假设不考虑一致维护和替…

【Redis】如何保证缓存和数据库的一致性

目录 背景问题思路 三个经典的缓存模式Cache-Aside读缓存写缓存为什么是删除旧缓存而不是更新旧缓存&#xff1f;为什么不先删除旧的缓存&#xff0c;然后再更新数据库&#xff1f; 延迟双删如何确保原子性 Read-Through/Write-ThroughRead-ThroughWrite-Through Write Behind …

Ubuntu22.04 下安装Curl库

1. apt 安装&#xff1a; sudo apt-get install curl 2. 官网压缩包&#xff1a; 下载地址&#xff1a;curl downloads wget https://curl.haxx.se/download/curl-7.78.0.tar.gz tar -xzvf curl-7.78.0.tar.gz cd curl-7.78.0 ./configure --with-openssl make sudo make i…