代理模式实现

news/2025/1/19 0:52:38/

一、概念:代理模式属于结构型设计模式。客户端不能直接访问一个对象,可以通过代理的第三者来间接访问该对象,代理对象控制着对于原对象的访问,并允许在客户端访问对象的前后进行一些扩展和处理;这种设置模式称为代理模式

2个点:

由代理对象去控制对原对象的访问(方法调用),客户端不直接对原对象进行访问,比如数据库的访问,不由Controller去控制;

在访问原对象时,可以有代理对象对原对象的功能(方法)进行扩展,比如访问数据库时,可以在代理对象里面进行开启事务和提交事务或回滚事务;

二、UML类图:

三、实现

1、静态代理实现

/*** Subject*/
public interface IUserMapperInterface {void save(String userName, String userNumber);
}

/*** RealSubject*/
public class UserMapperInterfaceImpl implements IUserMapperInterface {@Overridepublic void save(String userName, String userNumber) {System.out.println("执行insert语句,保存数据,并返回成功!");}
}

/*** 代理类*/
public class UserMapperInterfaceProxy implements IUserMapperInterface {private IUserMapperInterface userMapperInterface;public UserMapperInterfaceProxy(IUserMapperInterface userMapperInterface) {this.userMapperInterface = userMapperInterface;}@Overridepublic void save(String userName, String userNumber) {System.out.println("开启事务~~~");userMapperInterface.save(userName, userNumber);System.out.println("提交事务");}
}

/*** 客户端*/
public class Client {public static void main(String[] args) {String userName = "user name";String userNumber = "user number";UserMapperInterfaceProxy proxy = new UserMapperInterfaceProxy(new UserMapperInterfaceImpl());proxy.save(userName, userNumber);}
}

优缺点:

优点:可以在不修改目标类的前提下,对目标类进行扩展

缺点:

代码冗余,代理对象实现了和目标对象一致的接口,产生了一些冗余接口;

一旦目标接口中增加了新的方法,那么代理对象和真实接口实现都需要修改代码才行;

2、动态代理的实现:

动态代理与静态代理的区别:

静态代理在类加载完经已经加载JVM中,并且如有其他对象需要加载,需要修改代理对象;

动态代理只有在运行的时候才会去加载目标类, 当需要加载其他对象的时候,不需要修改,也就是动态代理可以代理任何对象Object;

动态代理有两种实现方式:JDK动态代理,和CGLib动态代理

  • JDK的动态代理,通过动态的生成一个接口对象的实现实例来实现对接口对象实例的功能扩展,在JDK的动态代理实现中,目标对象必须实现一个接口;
  • 通过反射包下的Proxy.newProxyInstance()方法实现

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;/*** 代理的工厂类*/
public class ProxyFactory {//需要代理的目标类private Object target;public ProxyFactory(Object target) {this.target = target;}public Object getTargetInstance() {//Proxy是反射包下(java.lang.reflect)的一个代理类,通过这个类的newProxyInstance(新建代理对象实例)来动态的加载一个代理目标对象,并返回//newProxyInstance方法需要传递三个参数,1、目标对象的类加载器, 2、目标对象实现的接口(如果有实现就有,没有实现就没有),3、目标对象需要执行的方法,相当于扩展静态代理中的那个save方法return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), new InvocationHandler() {//,目标对象被代理后需要执行的扩展方法,也有三个参数,1、目标对象,2、目标对象需要执行的方法实例,3、方法实例需要传递的参数@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("开启一个事务~!");//执行方法,执行的时候把目标对象和方法参数传递过去method.invoke(target, args);System.out.println("提交事务");//返回值是指目标执行的方法的返回值,如果没有就返回null,在静态代理中,目标方法save的返回值是void,因此这里返回nullreturn null;}});}
}

/*** 客户端*/
public class Client {public static void main(String[] args) {String userName = "user name";String userNumber = "user number";//使用静态代理的模式实现代理,此时在类加载的时候就会创建该对象UserMapperInterfaceProxy proxy = new UserMapperInterfaceProxy(new UserMapperInterfaceImpl());proxy.save(userName, userNumber);System.out.println("静态代理实现的类名:" + proxy.getClass().getName());//使用动态代理的模式实现代理功能,在类加载的时候不会创建对象,在运行过程中才会去创建对象并调用对象 的方法ProxyFactory proxyFactory = new ProxyFactory(new UserMapperInterfaceImpl());IUserMapperInterface targetInstance = (IUserMapperInterface) proxyFactory.getTargetInstance();targetInstance.save(userName, userNumber);System.out.println("动态代理实现的类名:" + targetInstance.getClass().getName());}
}
  • CGLib动态代理的实现

CGLib是一个第三方的代码生成类库,运行时在内存中生成一个子类继承目标对象,从而实现对目标对象的扩展;SpringAOP和Hibernate都是基于CGLib动态代理实现的,而CGlib又是通过封装ASM工具包去操作字节码的;

引入包:如果是Spring项目,则不需要单独引入,SpringCore里面已经包含该包

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

目标类:


/*** CGLib方式实现代理的目标类*/
public class CGLibTargetClass {public void targetMethod(String userName, String userNumber) {System.out.println("CGLibTargetClass targetMethod");System.out.println("保存用户信息:username=" + userName + ",usernumber=" + userNumber);}
}

代理类:实现MethodInterceptor接口,并重写interceptor方法,

public class CGLibProxyIntercept implements MethodInterceptor {@Overridepublic Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {System.out.println("保存用户信息前开启事务");System.out.println(objects.toString());System.out.println(o.getClass().getName());methodProxy.invokeSuper(o, objects);System.out.println("保存用户信息后提交事务");return o;}
}

代理模式的总结:

由于CGlib采用的是继承目标类的方式去进行增强的,因此对于final类或方法是不能被继承的,无法使用CGlib的方式进行代理;

在jdk1.8后,实际上JDK动态代理比CGLib动态代理效率高,但JDK动态代理需要目标类实现一个接口,因此如果对象有实现接口,用JDK动态代理,没实现 用CGLib动态代理;

有缺点:

在客户端与目标对象之间起到中介作用,保护目标对象

对目标对象进行扩展,将客户端与目标对象分离,解耦

增加系统复杂度


http://www.ppmy.cn/news/1564270.html

相关文章

电力场景红外测温图像绝缘套管分割数据集labelme格式2436张1类别

数据集格式&#xff1a;labelme格式(不包含mask文件&#xff0c;仅仅包含jpg图片和对应的json文件) 图片数量(jpg文件个数)&#xff1a;2436 标注数量(json文件个数)&#xff1a;2436 标注类别数&#xff1a;1 标注类别名称:["arrester"] 每个类别标注的框数&am…

前端web

学习笔记&#xff1a; 基本属性 color: 设置文本的颜色。代码&#xff1a;color: red;background-color: 设置元素的背景颜色。background-color: blue;font-size: 设置文本的大小font-size: 16px;font-family: 设置文本的字体font-family: Arial, sans-serif;text-align: 设…

使用 Vue.js 3 开发动态模块化组件:实现插件式表单系统

在现代前端开发中&#xff0c;模块化和可扩展性是开发复杂应用程序的核心目标。Vue.js 3 提供了很多强大的工具和功能&#xff0c;帮助我们实现这些目标。在本文中&#xff0c;我们将通过一个实际案例&#xff1a;构建动态模块化的插件式表单系统&#xff0c;深入了解如何高效利…

《Opencv》多对象模板匹配

目录 一、简介 二、模板匹配的基本原理 三、代码实现 四、结果展示 五、代码解析 1. 图像读取与预处理 2. 模板旋转 3. 模板匹配 4. 绘制矩形框 5. 结果显示 六、核心知识点 1. 模板匹配的局限性 2. 多角度模板匹配 3. 颜色与可视化 七、应用场景 八、总结 一、…

JUC Java并发编程 高级 学习大纲 动员

目录 口诀 锁 阿里巴巴开发规范 字节面试题 面试题 1 面试题 2 鼓舞 口诀 高内聚低耦合前提下 封装思想 线程 -- 操作 -- 资源类 判断、干活、通知防止虚假唤醒 &#xff0c;wait 方法要注意注意标志位 flag 可能是 volatile 的 锁 阿里巴巴开发规范 参考书 并发编程…

JavaWeb 前端基础 html + CSS 快速入门 | 018

今日推荐语 指望别人的救赎&#xff0c;势必走向毁灭——波伏娃 日期 学习内容 打卡编号2025年01月17日JavaWeb 前端基础 html CSS018 前言 哈喽&#xff0c;我是菜鸟阿康。 今天 正式进入JavaWeb 的学习&#xff0c;简单学习 html CSS 这2各前端基础部分&am…

抖音矩阵是什么

抖音矩阵是指在同一品牌或个人IP下&#xff0c;通过创建多个不同定位的抖音账号&#xff08;如主号、副号、子号等&#xff09;&#xff0c;形成一个有机的整体&#xff0c;以实现多维度、多层次的内容覆盖和用户互动。以下是关于抖音矩阵的详细介绍&#xff1a; 抖音矩阵的类…

PHP生产管理系统

生产管理系统&#xff1a;企业数字化转型的智慧引擎 &#x1f680; &#x1f4bb; 这是一款基于PHPLayuiuniapp框架&#xff0c;匠心独运的生产管理系统&#xff0c;专为推动企业向数字化、智能化转型而生。它是一套全面且高度定制化的解决方案&#xff0c;深度贴合各类生产企…