【代理模式】了解篇:静态代理 动态代理~

目录

1、什么是代理模式?

2、静态代理

3、动态代理

3.1 JDK动态代理类

3.2 CGLIB动态代理类

 4、JDK动态代理和CGLIB动态代理的区别?


1、什么是代理模式?

定义:

        代理模式就是为其他对象提供一种代理以控制这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象就可以在客户端和目标对象之间起到中介的作用。类似于租房:租户和房东之间现在不直接沟通,找了中介来联系,那么中介就是代理,这种模式就是代理模式。

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

2、静态代理

        在静态代理中,我们对目标对象的每个方法的增强都是手动完成的,非常不方便(比如接口一旦增加新方法,目标对象和代理对象都要进行修改)同时也麻烦(需要对于每个目标类都单独写一个代理类,类似于房东有很多套房子,每套房子都要找一个中介来代理),因此实际场景中静态代理的使用非常少。

        (1)从JVM的层面上讲:静态代理是在编译的时候就将接口、实现类、代理类等都变成了一个个实际的class文件。

模拟租户和房东之间的租房业务:

静态代理的使用步骤:

(1)创建一个接口和实现类——相当于房主现在要出租房;

//定义接口
public interface SellHouseService {void sell();
}

 不过房主有很多套房子,现在只想出租房子一;

//实现接口:具体的房东要出租房子
public class SellHouseOneService implements SellHouseService{@Overridepublic void sell() {System.out.println("SellHouseService...");}
}

(2)创建一个代理类也实现这个接口 ——房主将房子一出租的任务交给中介,不然中介也没有权限去给租户看房子。

//创建代理类
public class StaticProxy implements SellHouseService{//将目标对象注入到代理类(),然后在代理类的对应方法调用目标类中的对应方法private final SellHouseService sellHouseService;public StaticProxy(SellHouseService sellService){this.sellHouseService = sellService;}@Overridepublic void sell() {System.out.println("before...");sellHouseService.sell();System.out.println("after...");}
}

(3)主函数

public class Main {public static void main(String[] args) {//创建目标对象:房东SellHouseService service = new SellHouseOneService();//创建代理对象:中介SellHouseService proxy = new StaticProxy(service);proxy.sell();}
}

当我租第二套房子,第三套的时候也要找一个代理...非常麻烦。这个时候我们就使用动态代理了。

3、动态代理

相比于静态代理,动态代理更加灵活,我们不用再针对每个目标类都单独去创建一个代理类了。

小Tips

(1)动态代理主要有两种:JDK动态代理和CGLIB动态代理机制。

(2)其中,Spring AOP的实现就依赖了动态代理。

(3)从JVM角度讲:动态代理是在运行的时候动态生成类字节码,并加载到JVM中。

(4)在Java动态dialing机制中:InvocationHander接口和Proxy类是核心。

3.1 JDK动态代理类

JDK动态代理类的使用步骤:

(1)定义接口及实现类(同静态代理)

(2)自定义InvocationHandler并重写invoke方法,在invoke方法中调用被代理类的方法并定义一些处理逻辑;

//自定义InvokeHandler并重写invoke方法
public class JDKInvocationHandler implements InvocationHandler {//目标对象就是被代理对象private Object target;public JDKInvocationHandler(Object target){this.target = target;}//代理对象,要运行的方法,传递的实参@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//增强方法:前置处理//1、安全检查System.out.println("安全检查");//2、记录日志System.out.println("记录日志");//3.时间统计开始System.out.println("记录开始时间");//4、在invoke方法中会通过反射调用被代理类的方法Object retVal = method.invoke(target,args);//5、时间统计结束System.out.println("记录结束时间");return retVal;}
}

(3)创建一个代理类并使用。

public class Main {public static void main(String[] args) {//创建代理对象SellHouseService target = new SellHouseOneService();//创建一个代理类:通过被代理类、被代理实现的接口、方法调用处理器来创建SellHouseService proxy = (SellHouseService) Proxy.newProxyInstance(target.getClass().getClassLoader(),new Class[]{SellHouseService.class},new JDKInvocationHandler(target));proxy.sell();}
}

 小Tips

问:如何为Java对象创建一个代理对象?

        java.lang.reflect.Proxy类:提供了为对象产生代理对象的方法;

(1)Public static Object newProxyInstance(ClassLoader loader,Class<?> [ ] interfaces,InvocationHandler h),newProxyInstance作用是生成一个代理对象。返回值就是代理;

(2)这个方法一共有三个参数:loader类加载器,用于加载代理对象;interfaces:被代理类实现的一些接口;还有实现了InvocationHanldler接口的对象。

(3)JDK动态代理的问题:只能代理实现了接口的类:因此使用CGLIB动态代理来解决。

(4)通过Proxy 类的 newProxyInstance() 创建的代理对象在调用方法的时候,实际会调用到实现InvocationHandler 接口的类的 invoke()方法。 你可以在 invoke() 方法中自定义处理逻辑,比如在方法执行前后做什么事情。 

3.2 CGLIB动态代理类

        CGLIB(Code Generation Library)允许我们在运行的时候对字节码进行修改和动态生成。CGLIB允许我们通过继承的方式实现代理。CGLIB动态代理机制中MethodIntercepor接口Enhancer类是核心。在Spring AOP的设计中,如果目标对象实现了接口,则默认采用JDK动态代理,否则使用CGLIB动态代理。

CGLIB动态代理类的使用步骤:

(1)添加依赖

	    <dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>3.3.0</version></dependency>

(2)自定义MethodInterceptor并重写Intercept方法,intercept用于拦截增强被代理类的方法,和JDK动态代理中的Invoke方法类似;

public interface MethodInterceptor extends Callback{// 拦截被代理类中的⽅法public Object intercept(Object obj, java.lang.reflect.Method method, Object[] args,MethodProxy proxy) throws Throwable;
}
public class CGLIBInterceptor implements MethodInterceptor {//被代理对象private Object target;public CGLIBInterceptor(Object target){this.target = target;}@Overridepublic Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {//1.安全检查System.out.println("安全检查");//2.记录⽇志System.out.println("记录⽇志");//3.时间统计开始System.out.println("记录开始时间");//通过cglib的代理⽅法调⽤Object retVal = methodProxy.invoke(target, args);//4.时间统计结束System.out.println("记录结束时间");return retVal;}
}

(3)通过Enhancer类的create()创建代理类并使用。

public static void main(String[] args) {SellHouseService target= new SellHouseOneService();SellHouseService proxy= (SellHouseService) Enhancer.create(target.getClass(),new CGLIBInterceptor(target));proxy.sell();}

 4、JDK动态代理和CGLIB动态代理的区别?

(1)JDK 动态代理只能代理实现了接⼝的类或者直接代理接⼝,⽽ CGLIB 可以代
理未实现任何接⼝的类。
(2)CGLIB 动态代理是通过⽣成⼀个被代理类的⼦类来拦截被代理类的⽅法调
⽤,因此不能代理声明为 final。

(3)性能: ⼤部分情况都是 JDK 动态代理更优秀,随着 JDK 版本的升级,这个优势更
加明显 

 

 

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

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

相关文章

强引用和弱引用

什么是弱引用和强引用 强引用&#xff1a; JavaScript 中强引用&#xff1a;对象的引用在 JavaScript 中是强引用&#xff0c;也就是将一个引用对象通过变量或常量保存时&#xff0c;那么这个变量或常量就是强引用&#xff0c;这个对象就不会被回收。 弱引用&#xff1a; JavaS…

P1123 取数游戏(dfs,嘎嘎清晰)

1分析&#xff1a;这一题每个数是否选择会影响后面的选择情况&#xff0c;所以需要用一个数组来保存 所以状态为当前选到那个数&#xff0c;之前选的数的和以及之前每个数是否选了 之后直接搜索即可。尽管复杂度较高&#xff0c;但因为存在大量的不合法情况所以可以通过 时间复…

华为nat64配置

1.前期环境准备 环境拓扑 拓扑分为两个区域,左边为trust区域,使用IPv4地址互访,右边为untrust区域,使用IPv6地址互访 2.接口地址配置 pc1地址配置 pc2地址配置 FW接口配置 (1)首先进入防火墙配置界面 注:防火墙初始账号密码为user:admin,pwd:Admin@123,进入之后…

web攻击面试|网络渗透面试(三)

Web攻击大纲 常见Web攻击类型&#xff1a; SQL注入攻击&#xff1a;介绍SQL注入攻击的概念、原理和常见的攻击方式&#xff0c;如基于错误消息的注入、基于布尔盲注的注入等。解释攻击者如何利用SQL注入漏洞获取敏感信息或者对数据库进行恶意操作&#xff0c;并提供防御措施&a…

8.docker仓库

文章目录 Docker仓库本地私有仓库Docker HarborDocker harbor部署访问页面创建用户下载私有仓库镜像harbor同步 Docker仓库 本地私有仓库 ##先下载 registry 镜像docker pull registry##修改配置文件&#xff0c;在 daemon.json 文件中添加私有镜像仓库地址vim /etc/dock…

SQL-每日一题【1070. 产品销售分析 III】

题目 销售表 Sales&#xff1a; 产品表 Product&#xff1a; 编写一个 SQL 查询&#xff0c;选出每个销售产品 第一年 销售的 产品 id、年份、数量 和 价格。 结果表中的条目可以按 任意顺序 排列。 查询结果格式如下例所示&#xff1a; 示例 1&#xff1a; 解题思路 前置知…

【设计模式】 策略模式

策略模式&#xff08;Strategy Pattern&#xff09;是一种行为型设计模式&#xff0c;它定义了一系列算法&#xff0c;将每个算法封装起来&#xff0c;使它们可以相互替换&#xff0c;让客户端代码和算法的具体实现解耦。这样&#xff0c;客户端可以根据不同的需求选择不同的算…

Python爬虫的urlib的学习(学习于b站尚硅谷)

目录 一、页面结构的介绍  1.学习目标  2.为什么要了解页面&#xff08;html&#xff09;  3. html中的标签&#xff08;仅介绍了含表格、无序列表、有序列表、超链接&#xff09;  4.本节的演示 二、Urllib  1.什么是互联网爬虫&#xff1f;  2.爬虫核心  3.爬虫…

Visual Studio 2015编译器 自动生成 XXX_EXPORTS宏

XXX_EXPORTS宏 XXX_EXPORTS宏是由Visual Studio 2015编译器自动生成的。这个宏用于标识当前项目是一个导出符号的动态链接库&#xff08;DLL&#xff09;项目。在使用Visual Studio 2015创建Win32项目时&#xff0c;编译器会自动添加这个宏到项目的预定义宏中。 这个宏的作用…

【MySQL】复合查询

复合查询目录 一、基本查询二、多表查询三、自连接四、子查询4.1 单行子查询4.2 多行子查询4.3 多列子查询4.4 在from子句中使用子查询4.5 合并查询4.5.1 union4.5.2 union all 五、实战OJ 一、基本查询 --查询工资高于500或岗位为MANAGER的雇员&#xff0c;同时还要满足他们的…

web APIs-练习二

轮播图点击切换&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta http-equiv"X-UA-Compatible" content"IEedge" /><meta name"viewport" content"…

LLaMA模型论文《LLaMA: Open and Efficient Foundation Language Models》阅读笔记

文章目录 1. 简介2.方法2.1 预训练数据2.2 网络架构2.3 优化器2.4 高效的实现 3.论文其余部分4. 参考资料 1. 简介 LLaMA是meta在2023年2月开源的大模型&#xff0c;在这之后&#xff0c;很多开源模型都是基于LLaMA的&#xff0c;比如斯坦福大学的羊驼模型。 LLaMA的重点是比…

从Vue层面 - 解析发布订阅模式和观察者模式区别

目录 前言一、发布订阅模式什么是发布订阅模式&#xff1f;应用场景 二、观察者模式1&#xff09;什么是观察者模式&#xff1f;2&#xff09;应用场景3&#xff09;vue中的观察者模式观察者&#xff08;订阅者&#xff09; - Watcher目标者&#xff08;发布者&#xff09; - D…

剑指 Offer 46.! 把数字翻译成字符串(动态规划,青蛙跳台问题的变形)

剑指 Offer 46. 把数字翻译成字符串 中等 588 相关企业 给定一个数字&#xff0c;我们按照如下规则把它翻译为字符串&#xff1a;0 翻译成 “a” &#xff0c;1 翻译成 “b”&#xff0c;……&#xff0c;11 翻译成 “l”&#xff0c;……&#xff0c;25 翻译成 “z”。一个数字…

MATLAB算法实战应用案例精讲-【深度学习】预训练模型-Subword

目录 前言 Subword 1. Subword介绍 分词器是做什么的? 为什么需要分词? 分词方法

rtthread的idle线程不应该长时间堵塞

RT-Thread是一个实时嵌入式操作系统&#xff0c;它的空闲线程&#xff08;Idle Thread&#xff09;是在系统中没有其他任务需要执行时运行的线程。空闲线程通常用于执行一些低优先级的任务或者进行系统的休眠等操作。 RT-Thread的空闲线程不能在线程中堵塞的原因是为了确保系统…

STM32CubeIDE(串口)

目录 一、轮询模式 1.1 配置USART2为异步模式 1.2 500ms发送一次消息 1.3 通信结果 1.4 串口控制LED 二、中断收发 2.1 开启中断 2.2 中断发送接收 2.2.1 中断发送只需要调用接口 2.2.2 中断接收 2.3 实验结果 三、DMA模式与收发不定长数据 3.1 DMA通道配置 3.2 DMA…

设计模式-命令模式在Java中的使用示例-桌面程序自定义功能键

场景 欲开发一个桌面版应用程序&#xff0c;该应用程序为用户提供了一系列自定义功能键&#xff0c;用户可以通过这些功能键来实现一些快捷操作。 用户可以将功能键和相应功能绑定在一起&#xff0c;还可以根据需要来修改功能键的设置&#xff0c;而且系统在未来可能还会增加…

Python与OpenCV环境中,借助SIFT、单应性、KNN以及Ransac技术进行实现的图像拼接算法详细解析及应用

一、引言 在当今数字化时代,图像处理技术的重要性不言而喻。它在无人驾驶、计算机视觉、人脸识别等领域发挥着关键作用。作为图像处理的一个重要部分,图像拼接算法是实现广阔视野图像的重要手段。今天我们将会讲解在Python和OpenCV环境下,如何使用SIFT、单应性、KNN以及Ran…

使用 OpenCV 进行图像模糊度检测(拉普拉斯方差方法)

写在前面 工作中遇到&#xff0c;简单整理人脸识别中&#xff0c;对于模糊程度较高的图像数据&#xff0c;识别率低&#xff0c;错误率高。虽然使用 AdaFace 模型&#xff0c;对低质量人脸表现尤为突出。但是还是需要对 模糊程度高的图像进行丢弃处理当前通过阈值分类&#xff…