代理模式:控制访问的设计模式

代理模式:控制访问的设计模式

什么是代理模式?

代理模式是一种常见的设计模式,它允许通过代理对象来控制对真实对象的访问。代理模式的主要目的是在不改变原始对象的情况下,提供额外的功能或控制访问。

为什么要使用代理模式?

代理模式有以下几个主要的应用场景:

  • 访问控制:代理模式可以限制对真实对象的直接访问,只有通过代理对象才能访问真实对象。这样可以实现对真实对象的访问控制,例如权限验证、身份验证等。
  • 增加额外功能:代理模式可以在不修改真实对象的情况下,为其增加额外的功能。代理对象可以在调用真实对象的方法前后执行一些额外的操作,例如日志记录、性能监控、缓存等。
  • 远程访问:代理模式可以实现远程访问,即通过代理对象访问位于不同地址空间的真实对象。这对于分布式系统或跨网络的应用程序非常有用。

代理模式的两种分类

静态代理

静态代理是在编译时就已经确定代理对象和真实对象的关系。代理对象和真实对象实现相同的接口或继承相同的父类,代理对象持有真实对象的引用,并在调用真实对象的方法前后执行一些额外的操作。

优点: 简单易懂

缺点: 需要为每个真实对象编写一个代理类,当真实对象较多时,会导致代码冗余

案例: 一个简单的日志记录功能

假设我们有一个 UserService 接口和一个实现类 UserServiceImpl,它提供了用户管理的一些基本操作方法,如添加用户、删除用户等。现在我们需要在每个方法执行前后记录日志,例如:在方法执行前,打印 “Before” 的日志;在方法执行后,打印 “After” 的日志。
在这里插入图片描述

// UserService 接口
public interface UserService {void addUser();void deleteUser();
}// 实现类 UserServiceImpl
public class UserServiceImpl implements UserService {@Overridepublic void addUser() {System.out.println("添加用户...");}@Overridepublic void deleteUser() {System.out.println("删除用户...");}
}// 静态代理类
public class UserServiceProxy implements UserService {private UserService userService;public UserServiceProxy(UserService userService) {this.userService = userService;}@Overridepublic void addUser() {System.out.println("Before...");userService.addUser();System.out.println("After...");}@Overridepublic void deleteUser() {System.out.println("Before...");userService.deleteUser();System.out.println("After...");}// 其他方法同样的方式实现
}

接下来,我们可以使用代理类来代替真实对象进行操作。

public class Client {public static void main(String[] args) {// 静态代理UserServiceProxy proxy = new UserServiceProxy(new UserServiceImpl());proxy.addUser();System.out.println("------------------");proxy.deleteUser();}
}

结果:
在这里插入图片描述

动态代理

动态代理是一种在运行时动态生成代理类的代理模式。它可以在不修改原始类的情况下,为原始类提供额外的功能或控制访问。在Java中,有两种常见的动态代理方式:JDK动态代理和CGLIB动态代理

JDK动态代理

JDK动态代理是通过Java的反射机制实现的。它要求被代理的类必须实现一个接口。JDK动态代理提供了一个Proxy类和一个InvocationHandler接口,通过这两个类可以动态生成代理类。

案例: 简单的日志记录功能

定义一个接口 UserService,它提供了用户管理的一些基本操作方法。

public interface UserService {void addUser();void deleteUser();
}

真实的用户服务类UserServiceImpl,它实现了 UserService 接口。

public class UserServiceImpl implements UserService {@Overridepublic void addUser() {System.out.println("添加用户...");}@Overridepublic void deleteUser() {System.out.println("删除用户...");}
}

通过Proxy类,创建代理对象UserServiceProxy对目标对象的方法进行拦截和增强

public class UserServiceProxy {private Object target;public UserServiceProxy(Object target) {this.target = target;}/*** target:目标对象,即要被代理的对象。* getProxyInstance():该方法返回一个代理对象,该代理对象实现了目标对象所实现的接口,并在方法调用前后执行额外的逻辑。* Proxy.newProxyInstance():这是Java提供的创建代理对象的方法。它接受三个参数:类加载器、目标对象实现的接口数组和一个InvocationHandler对象。* InvocationHandler:这是一个接口,用于定义代理对象的方法调用处理逻辑。在invoke()方法中,我们可以在目标方法调用前后执行额外的逻辑。*/public  Object getProxyInstance(){return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),new InvocationHandler() {@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println(gdk代理模式---开始);Object invoke = method.invoke(target, args);System.out.println(gdk代理模式---结束);return invoke;}});}
}

最后,我们可以使用 Proxy 类的 newProxyInstance 方法来创建代理对象。

public class Client {public static void main(String[] args) {// JDK动态代理UserService proxy = (UserService) new UserServiceProxy(new UserServiceImpl()).getProxyInstance();proxy.addUser();System.out.println("-----------------");proxy.deleteUser();}
}

在这里插入图片描述

CGLIB动态代理

CGLIB动态代理是通过继承被代理类来实现的,它不要求被代理的类实现接口。CGLIB动态代理使用了字节码生成库来生成代理类。

案例:简单的日志记录功能

定义一个类 UserService,它提供了用户管理的一些基本操作方法。

public interface UserService {void addUser();void deleteUser();// 带返回结果int getUserCount();
}

真实的用户服务类UserServiceImpl,它实现了 UserService 接口。

public class UserServiceImpl implements UserService {@Overridepublic void addUser() {System.out.println("添加用户...");}@Overridepublic void deleteUser() {System.out.println("删除用户...");}// 带返回结果@Overridepublic int getUserCount() {System.out.println("查询用户数量...");return 66;}
}

一个代理类 UserServiceProxy ,它实现了MethodInterceptor 接口。

public class UserServiceProxy implements MethodInterceptor {// 维护一个目标对象private Object target;// 构造器,传入一个被代理的对象public UserServiceProxy(Object target) {this.target = target;}// 返回一个代理对象,target的代理对象public Object getProxyInstance() {// 1. 创建一个工具类Enhancer enhancer = new Enhancer();// 2. 设置父类enhancer.setSuperclass(target.getClass());// 3. 设置回调函数enhancer.setCallback(this);// 4. 创建子类对象,即代理对象return enhancer.create();}// 重写 intercept 方法,会调用目标对象的方法@Overridepublic Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {System.out.println("cglib代理模式---开始");Object returnVal = method.invoke(target, objects);System.out.println("cglib代理模式---结束");return returnVal;}
}

最后,我们可以使用UserServiceProxygetProxyInstance方法创建代理对象实例,调用方法。

public static void main(String[] args) {// 创建目标对象UserService userService = new UserServiceImpl();// 获取代理对象,并将目标对象传递到代理对象UserService proxyInstance = (UserService) new UserServiceProxy(userService).getProxyInstance();// 执行代理对象的方法,触发intecept方法,从而实现目标对象的调用proxyInstance.addUser();System.out.println("-------------");int userCount = proxyInstance.getUserCount();System.out.println(userCount);}

在这里插入图片描述

jdk动态代理和cglib动态代理的区别
  1. 实现方式:jdk动态代理是通过反射实现的,而cglib动态代理是通过继承目标类来实现的。
  2. 目标类限制:jdk动态代理要求目标类必须要实现接口,而cglib动态代理则没有这个限制。
  3. 性能:jdk动态代理相对于cglib动态代理来说,因为实现方式不同,生成的代理类的效率会低一些
  4. 对象类型:jdk动态代理只能代理实现了接口的类,cglib通过继承实现,不能代理 final 类
  5. 依赖库:jdk动态代理是Java自带的库,不需要额外的依赖,而cglib动态代理需要依赖cglib库

总结

代理模式是一种非常有用的设计模式,它可以实现访问控制、增加额外功能和远程访问。静态代理在编译时确定代理对象和真实对象的关系,而动态代理在运行时动态生成代理对象。动态代理又分为jdk动态代理和cglib动态代理,分别基于接口和类来实现代理功能。根据具体的需求和场景,选择适合的代理模式可以提高代码的可维护性和灵活性。
区别:

  • 与适配器模式的区别适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。
  • 与装饰器模式的区别装饰器模式为了增强功能,而代理模式是为了加以控制。

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

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

相关文章

MFC使用png做背景图片

在MFC中使用png图片作为背景&#xff0c;你需要使用GDI库。以下是一个简单的示例&#xff1a; 首先&#xff0c;你需要在你的项目中包含GDI头文件和库。在你的stdafx.h&#xff08;或者你的项目预编译头文件&#xff09;中添加以下代码&#xff1a; #include <GdiPlus.h&g…

2023年Q2京东环境电器市场数据分析(京东数据产品)

今年Q2&#xff0c;环境电器市场中不少类目表现亮眼&#xff0c;尤其是以净水器、空气净化器、除湿机等为代表的环境健康电器。此外&#xff0c;像冷风扇这类具有强季节性特征的电器也呈现出比较好的增长态势。 接下来&#xff0c;结合具体数据我们一起来分析Q2环境电器市场中…

【LeetCode】198.打家劫舍

题目 你是一个专业的小偷&#xff0c;计划偷窃沿街的房屋。每间房内都藏有一定的现金&#xff0c;影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统&#xff0c;如果两间相邻的房屋在同一晚上被小偷闯入&#xff0c;系统会自动报警。 给定一个代表每个房屋存放…

所有流的知识都有,IO流原理及流的分类

1、Java IO流原理 I/O是Input/Output的缩写&#xff0c; I/O技术是非常实用的技术&#xff0c;用于处理设备之间的数据传输。如读/写文件&#xff0c;网络通讯等。 Java程序中&#xff0c;对于数据的输入/输出操作以”流(stream)” 的方式进行。java.io包下提供了各种“流”类…

微信小程序监测版本更新

在index.js里面 不放到app.js里面是因为有登录页面&#xff0c;在登录页面显示更新不太友好 onShow() {const updateManager wx.getUpdateManager()// 请求完新版本信息的回调updateManager.onCheckForUpdate(res > {if (res.hasUpdate) {// 新版本下载成功updateManage…

【OnnxRuntime】在linux下编译并安装C++版本的onnx-runtime

目录 安装C接口的onnx-runtime安装依赖项&#xff1a;下载源文件构建ONNX Runtime安装ONNX Runtime 安装C接口的onnx-runtime 安装依赖项&#xff1a; 安装CMake&#xff1a;可以通过包管理器&#xff08;如apt、yum等&#xff09;安装CMake。 安装C编译器&#xff1a;确保系…

WPF实战学习笔记25-首页汇总

注意&#xff1a;本实现与视频不一致。本实现中单独做了汇总接口&#xff0c;而视频中则合并到国todo接口当中了。 添加汇总webapi接口添加汇总数据客户端接口总数据客户端接口对接3首页数据模型 添加数据汇总字段类 新建文件MyToDo.Share.Models.SummaryDto using System;…

【JavaEE初阶】Servlet (二) Servlet中常用的API

文章目录 HttpServlet核心方法 HttpServletRequest核心方法 HttpServletResponse核心方法 Servlet中常用的API有以下三个: HttpServletHttpServletRequestHttpServletResponse HttpServlet 我们写 Servlet 代码的时候, 首先第一步就是先创建类, 继承自 HttpServlet, 并重写其…

以科技创新引领短交通行业发展,九号公司重磅新品亮相巴塞罗那MWC

2月27日&#xff0c;以“时不我待(VELOCITY) - 明日科技&#xff0c;将至已至”为主题的2023世界移动通信大会&#xff08;Mobile World Congress&#xff0c;以下简称MWC&#xff09;在西班牙巴塞罗那举办&#xff0c;全球创新短交通领军企业九号公司参加了大会。现场&#xf…

Pytorch(三)

一、经典网络架构图像分类模型 数据预处理部分: 数据增强数据预处理DataLoader模块直接读取batch数据 网络模块设置: 加载预训练模型&#xff0c;torchvision中有很多经典网络架构&#xff0c;可以直接调用注意别人训练好的任务跟咱们的并不完全一样&#xff0c;需要把最后…

2023年FPGA好就业吗?

FPGA岗位有哪些&#xff1f; 从芯片设计流程来看&#xff0c;FPGA岗位可以分四类 产品开发期&#xff1a;FPGA系统架构师 芯片设计期&#xff1a;数字IC设计工程师、FPGA开发工程师 芯片流片期&#xff1a;FPGA验证工程师 产品维护期&#xff1a;FAE工程师 从行业上来说&#x…

6、Kubernetes核心技术 - Pod

目录 一、概述 二、Pod机制 2.1、共享网络 2.2、共享存储 三、Pod资源清单 四、 Pod 的分类 五、Pod阶段 六、Pod 镜像拉取策略 ImagePullBackOff 七、Pod 资源限制 八、容器重启策略 一、概述 Pod 是可以在 Kubernetes 中创建和管理的、最小的可部署的计算单元。P…

时序预测 | Python实现NARX-DNN空气质量预测

时序预测 | Python实现NARX-DNN空气质量预测 目录 时序预测 | Python实现NARX-DNN空气质量预测效果一览基本介绍研究内容程序设计参考资料效果一览 基本介绍 时序预测 | Python实现NARX-DNN空气质量预测 研究内容 Python实现NARX-DNN空气质量预测,使用深度神经网络对比利时空气…

有道OCR图文识别整合SpringBoot

背景需求, 官方SDK,在SpringBoot项目中过于臃肿,需要引入的Jar包过多, 在SpringBoot中, 本文使用SpringBoot中的RestTemplate对象进行请求接口 案例代码如下 package com.example.demo2.Test;import com.example.demo2.Test.Ocr.OcrResponse; import org.springframework.h…

基于51单片机和proteus的加热洗手器系统设计

此系统是基于51单片机和proteus的仿真设计&#xff0c;功能如下&#xff1a; 1. 检测到人手后开启出水及加热。 2. LED指示加热出水及系统运行状态。 功能框图如下&#xff1a; Proteus仿真界面如下&#xff1a; 下面就各个模块逐一介绍&#xff0c; 模拟人手检测模块 通过…

mysql 非definer用户如何查看存储过程定义

当我们创建存储过程时&#xff0c;如果没有显示指定definer&#xff0c;则会默认当前用户为该sp的definer&#xff0c;如果没有相关授权&#xff0c;则其他用户是看不了这个sp的。 比如用户zhenxi1拥有如下权限&#xff1a; 它拥有对dev_nacos库的查询权限&#xff0c;这个时候…

Xamarin.Android实现加载中的效果

目录 1、说明2、代码如下2.1 图1的代码2.1.1、创建一个Activity或者Fragment&#xff0c;如下&#xff1a;2.1.2、创建Layout2.1.3、如何使用 2.2 图2的代码 4、其他补充4.1 C#与Java中的匿名类4.2 、其他知识点 5、参考资料 1、说明 在实际使用过程中&#xff0c;常常会用到点…

【ChatGPT辅助学Rust | 基础系列 | 基础语法】变量,数据类型,运算符,控制流

文章目录 简介&#xff1a;一&#xff0c;变量1&#xff0c;变量的定义2&#xff0c;变量的可变性3&#xff0c;变量的隐藏 二、数据类型1&#xff0c;标量类型2&#xff0c;复合类型 三&#xff0c;运算符1&#xff0c;算术运算符2&#xff0c;比较运算符3&#xff0c;逻辑运算…

python_在K线找出波段_01_找出所有转折点

目录 写在前面&#xff1a; 需要考虑的几种K线图情况&#xff1a; 寻找所有转折点逻辑&#xff1a; 代码&#xff1a; 寻找转折点方法&#xff1a;找到第一个转折点就返回 循环寻找峰谷方法 主方法 调用计算&#xff1a; 返回&#xff1a; 在【PyQt5开发验证K线视觉想…

数字图像处理(番外)图像增强

图像增强 图像增强的方法是通过一定手段对原图像附加一些信息或变换数据&#xff0c;有选择地突出图像中感兴趣的特征或者抑制(掩盖)图像中某些不需要的特征&#xff0c;使图像与视觉响应特性相匹配。 图像对比度 图像对比度计算方式如下&#xff1a; C ∑ δ δ ( i , j …