代理模式--【学习笔记】
创始人
2024-04-22 10:56:38
0

什么是代理模式?
代理是一种模式,提供了对目标对象的间接访问方式,即通过代理对象访问目标对象.如此便于在目标实现的基础上增加额外的功能操作,以满足自身的业务需求.

在这里插入图片描述

代理模式又分为静态代理,动态代理

静态代理模式

编写代理类, 要求: 代理类与目标类实现相同的接口, 代理类维护一个目标类对象, 目标方法由目标对象完成, 代理类作为额外的功能

图示:
在这里插入图片描述

代码演示:
目标类接口

/*** 工作接口*/
public interface Work {//工作的方法public int work();
}

目标类:

/*** 目标类*/
public class Worker implements  Work{@Overridepublic int work() {System.out.println("工人正在工作...");return 0;}
}

代理类:

/*** 代理类  代理的Worker类*   能够代理实现Work接口所有的实现类*/
public class AWorkerProxy implements Work {//目标类对象private Work  worker;  // 组合//构造public AWorkerProxy(Work  worker){this.worker = worker;}//set方式@Overridepublic int work() {System.out.println("验证身份...");System.out.println("打上班卡");//调用目标类的目标方法int rs = worker.work();System.out.println("打下班卡");return rs;}
}

优点:
1.实现了不改变目录类前提, 增强目录的类的方法
2相对于继承来实现: 静态代理不需要使用继承, 节省类继承资源, 静态代理增强的某一接口所有实现类, 继承方式, 只能增强某一个类

缺点:
1.只能增强某一接口, 如果要增强多个不同接口的实现类, 创建多个代理类, 代理类暴增
2. 代码重复
3. 增强代码直接写在代理类方法中, 硬编码, 后期修改, 违背开闭原则

动态代理:基于JDK提供: Proxy

可以解决静态代理只能增强某一个接口,代码重复的问题;

首先可以看一下创建一个对象的过程

  1. 先编写Person.java
  2. 编译Person.java,生成Person.class 使用 javac
  3. 通过类加载器, 把Person.class 加载到内存
  4. 运行代码: java --> main(){new Person();}

图示:
在这里插入图片描述

根据图中的步骤2,执行java命令会启动JVM将字节码文件加载进内存,然后再根据Class对象来创建对象

图示:
在这里插入图片描述
顺序反过来看,如果我们想得到一个代理类对象,就需要有一个代理类的Class对象。
构造这个Class对象的思路:
1.首先我们要根据目标类的接口来创建一个架构一样的Class对象(空壳架构,JDK提供了Proxy类实现这个功能)
2.再将目标类中要增强的方法,增强类(用来给目标增加功能)中的方法整合在一起
3.最后再更具这个代理Class对象,创建实例

在这里插入图片描述
(图中未加上增强类的方法)

内部设计思路:
每次调用代理对象的方法都会调用invoke(),且invoke()的返回值就是代理方法的返回值。
在这里插入图片描述

代码演示:

 private  Object getProxy(Object target,Advice advice) throws  Exception{//第一步:通过Proxy类,创建一个和目标类接口一样的架构的Class对象//参数1:类加载器  参数2:目标类的接口Class proxyClass = Proxy.getProxyClass(target.getClass().getClassLoader(),target.getClass().getInterfaces());//创建对象需要用到构造方法,但是Proxy生成的代理类对象的无参构造方法是私有的//只能调用调用下面的带参方法来创建实例Constructor constructor = proxyClass.getDeclaredConstructor(InvocationHandler.class);//创建代理类实例Object proxy = constructor.newInstance(new InvocationHandler(){//method调用方法//proxy: 代理类对象//args: 方法参数@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//第二步:将目标类中方法,和增强类中的方法组合在一起,形成一个新的方法advice.before();//调用目标对象的目标方法Object rs =  method.invoke(target,args);advice.after();return rs;}});//第三步:返回代理类对象(经过增强的代理类)return proxy;}

注意:newInstance方法,返回指定接口的实体类,必须用接口来接收返回值对象,不能用实现类来接收。不然会报错:
为什么这么做?个人理解:
代理类是根据目标类的接口而生成的,可以认为它和这个目标类没有一点关系。如果把这个代理类看作是目标类接口的一个子类,那么目标类和代理类都是接口的子类,他们之间是不能进行强转的,所以不能用实现类来接收返回值。

在这里插入图片描述

优点:

  1. 不改变目标类的代码, 对目标类的方法增强
  2. 代理各种不同的接口

缺点:

  1. 违背开闭原则: 增强代码虽然封装在一个一个增强类中,但是需要在invoke()方法,调用增强类的方法, 硬编码, 修改,需要修改invoke()方法
  2. 要求目标类必须实现某个接口, 基于接口的代理

解决方案:

解决1的方案: 使用配置文件/注解 替换硬编码 --> AOP(底层基于动态代理实现)

解决2的方案: 动态代理的第二种实现: 第三方: cglib, 代理类是目标类的子类, 这个类可以实现接口,也可以不实现接口

相关内容

热门资讯

【MySQL】锁 锁 文章目录锁全局锁表级锁表锁元数据锁(MDL)意向锁AUTO-INC锁...
【内网安全】 隧道搭建穿透上线... 文章目录内网穿透-Ngrok-入门-上线1、服务端配置:2、客户端连接服务端ÿ...
GCN的几种模型复现笔记 引言 本篇笔记紧接上文,主要是上一篇看写了快2w字,再去接入代码感觉有点...
数据分页展示逻辑 import java.util.Arrays;import java.util.List;impo...
Redis为什么选择单线程?R... 目录专栏导读一、Redis版本迭代二、Redis4.0之前为什么一直采用单线程?三、R...
【已解决】ERROR: Cou... 正确指令: pip install pyyaml
关于测试,我发现了哪些新大陆 关于测试 平常也只是听说过一些关于测试的术语,但并没有使用过测试工具。偶然看到编程老师...
Lock 接口解读 前置知识点Synchronized synchronized 是 Java 中的关键字,...
Win7 专业版安装中文包、汉... 参考资料:http://www.metsky.com/archives/350.htm...
3 ROS1通讯编程提高(1) 3 ROS1通讯编程提高3.1 使用VS Code编译ROS13.1.1 VS Code的安装和配置...
大模型未来趋势 大模型是人工智能领域的重要发展趋势之一,未来有着广阔的应用前景和发展空间。以下是大模型未来的趋势和展...
python实战应用讲解-【n... 目录 如何在Python中计算残余的平方和 方法1:使用其Base公式 方法2:使用statsmod...
学习u-boot 需要了解的m... 一、常用函数 1. origin 函数 origin 函数的返回值就是变量来源。使用格式如下...
常用python爬虫库介绍与简... 通用 urllib -网络库(stdlib)。 requests -网络库。 grab – 网络库&...
药品批准文号查询|药融云-中国... 药品批文是国家食品药品监督管理局(NMPA)对药品的审评和批准的证明文件...
【2023-03-22】SRS... 【2023-03-22】SRS推流搭配FFmpeg实现目标检测 说明: 外侧测试使用SRS播放器测...
有限元三角形单元的等效节点力 文章目录前言一、重新复习一下有限元三角形单元的理论1、三角形单元的形函数(Nÿ...
初级算法-哈希表 主要记录算法和数据结构学习笔记,新的一年更上一层楼! 初级算法-哈希表...
进程间通信【Linux】 1. 进程间通信 1.1 什么是进程间通信 在 Linux 系统中,进程间通信...
【Docker】P3 Dock... Docker数据卷、宿主机与挂载数据卷的概念及作用挂载宿主机配置数据卷挂载操作示例一个容器挂载多个目...