深入理解静态与动态代理设计模式:从理论到实践

embedded/2025/3/16 20:22:53/

静态代理设计模式

1.为什么需要代理设计模式?

  • javaEE分层开发中,哪个层次对于我们来讲最重要

DAO---->Service---->Controller

JavaEE分层中,最为重要的是Service层

  •  Service层包含了那些代码

Service层=核心功能(几十行 上百代码) +额外功能(附加功能)

1.核心功能

   业务运算

   DAO调用

2.额外功能

    1.不属于业务

    2.可有可无

    3.代码量很小

事务、日志、性能

  •  额外功能书写在Service层中好不好?

Service层调用者角度(Controller):需要再Service层中书写额外功能

                软件设计者:Service层不需要额外功能

 现实生活中的解决方案

对话例子:

场景 1:直接找房东租房(无代理模式)

房客:你好,我想租你的房子,能带我看看吗?

房东:额……我只负责签合同,广告、带看这些不归我管,你自己去找吧。

房客:(无奈)这太麻烦了,我得自己找房源、安排时间,还可能被忽悠。

场景 2:通过中介租房(静态代理)

房客:你好,我想租房,你能帮我介绍吗?

中介:当然!我这有很多房源信息,还能带你去看房,满意的话就签合同。

(中介带房客去看房后……)

房客:这房子不错,我想租。

中介:好,那我帮你联系房东签合同吧。

房东:好的,我们来签合同吧。(完成租房)

场景 3:更换中介(动态代理)

房客:(租房体验不满意)这个中介效率太低了,我想换个更好的。

中介公司:没问题!我们这有多个中介,你可以换一个更专业的。

(更换代理后,房客又顺利租到了房子)

 2.代理设计模式

1.1概念

通过代理类:为原始类(目标)增加额外功能

好处:利于原始类(目标的维护)

 1.2名词解释

1.目标类 原始类
指的是 业务类(核心功能--->业务运算 DAO调用)

2.目标方法 原始方法

目标类(原始类)中的方法 就是目标方法(原始方法)

3.额外功能(附加功能)

日志、事务、性能

 1.3代理开发的核心要素

代理类=目标类(原始类)+额外功能+原始类(目标类)实现相同的接口

房东---->public interface UserService{

                m1方法

                m2方法

}        

                UserServiceImpl implements UserService{

                        m1---->业务调用 DAO调用

                        m2

                }

                UserServiceProxy implements UserService

 1.4编码

静态代理:为每一个原始类,手工编写一个代理类

java">public class UserServiceProxy implements UserService{private UserServiceImpl userService=new UserServiceImpl();@Overridepublic void register(User user) {System.out.println("---log");userService.register(user);}@Overridepublic void login(String name, String password) {System.out.println("---log");userService.login(name, password);}
}

1.5静态里存在的问题

1.静态类文件数量过多,不利于项目管理

UserServiceImpl UserServiceProxy

OrderServiceImpl OrderServiceProxy

2.额外功能维护性差

代理类中 额外功能修改复杂

 Spring动态代理

概念:通过代理类为原始类(目标类增加额外功能)

好处:利于原始类(目标类的维护)

 搭建开发环境

java"><dependency><groupId>org.springframework</groupId><artifactId>spring-aop</artifactId><version>5.2.14.RELEASE</version></dependency><dependency><groupId>org.aspectj</groupId><artifactId>aspectjrt</artifactId><version>1.8.8</version></dependency><dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.8.3</version></dependency>

Spring动态代理开发步骤

1.创建原始对象(目标对象)

java">public class UserServiceImpl implements UserService{@Overridepublic void register(User user) {System.out.println("用户注册成功");}@Overridepublic void login(String name, String password) {System.out.println("用户登录成功");}
}
<bean id="userService" class="proxy.UserServiceImpl"></bean>

2.额外功能

 MethodBeforeAdvice接口

额外功能书写在接口的视线中,运行在原始方法执行之前运行额外功能

java">public class Before implements MethodBeforeAdvice {/*作用:需要把原始方法之前运行的额外功能 写在before方法中*/@Overridepublic void before(Method method, Object[] objects, Object o) throws Throwable {System.out.println("------额外功能开发--------");}
}

 3.定义切入点

切入点:额外功能加入的位置

目的:由程序员根据自己的需要,决定额外功能加入倒哪个原始方法中

rigister

login

简单测试:所有方法都作为切入点 加入额外功能

java">  <aop:config>
<!--        所有的方法 都作为切入点 加入额外功能 login register--><aop:pointcut id="pc" expression="execution(* *(..))"/></aop:config>

4.组装(2.3整合)

java"><aop:config>
<!--        所有的方法 都作为切入点 加入额外功能 login register--><aop:pointcut id="pc" expression="execution(* *(..))"/><aop:advisor advice-ref="before" pointcut-ref="pc"></aop:advisor></aop:config>

5.调用

目的:获得Spring工厂创建的动态代理对象,并进行调用

ApplicationContext applicaton=new ClassPathXmlApplication("/applicationContext.xml");

注意:

        1.Spring工厂通过原始对象的id值获得是代理对象

        2.获得代理对象后,可以通过生命接口类型,进行对象存储

UserService userService=(UserService)application.getBean("userService");

application.getBean("userService")

 动态代理细节分析

1.Spring创建的动态代理类在哪里?

Spring框架在运行时,通过动态字节码奇数,在JVM创建,运行在JVM内部,等程序结束后,回合JVM一起消失

动态字节码技术

通过第三方动态字节码框架,在JVM中创建对应类的字节码,进而创建对象,当虚拟机结束,动态字节码跟着消失

结论:动态代理不需要定义类文件,都是JVM运行过程中动态创建的,所以不会造成静态代理,类文件数量过多,影响项目管理问题

2.动态代理编程简化代理开发

在额外功能不改变的情况下,创建其它目标类(原始类)的代理对象时,只需要指定原始(目标对象)即可。

3.动态代理可维护性大大增强

(当前代理不想用 直接替换即可)

java"><bean id="userService" class="proxy.UserServiceImpl"></bean><bean id="before" class="dynamic.Before"></bean><bean id="before2" class="dynamic.Before2"></bean><aop:config>
<!--        所有的方法 都作为切入点 加入额外功能 login register--><aop:pointcut id="pc" expression="execution(* *(..))"/><aop:advisor advice-ref="before2" pointcut-ref="pc"></aop:advisor></aop:config>

MethodBeforeAdvice详解

接口作用:额外功能在原始方法执行之前,进行额外功能操作
public void before(Method method, Object[] objects, Object o) throws Throwable {System.out.println("------额外功能开发--------");
}
Object O:额外对象所增加给的原始对象
Object []objects:所增加切入点方法的参数
Method method:额外功能所增加的所在的方法 切入点

before方法在实战中,如何使用
根据需要使用,不一定用到,可以能不用

---Spring学习笔记(跟整理) 来源:076_动态代理细节分析3_哔哩哔哩_bilibili


http://www.ppmy.cn/embedded/173151.html

相关文章

Protobuf 学习与实践

Protobuf 学习与实践 Protobuf 学习与实践一、安装 Protobuf1. 在 Ubuntu 上下载 Protobuf 编译器1.1 下载地址与版本选择1.2 安装依赖1.3 下载与解压 2. 安装 Protobuf2.1 配置与编译2.2 增大 Swap 分区&#xff08;若需要&#xff09;2.3 配置环境变量&#xff08;若自定义安…

深入理解C/C++堆数据结构:从原理到实战

一、堆的本质与特性 1.1 什么是堆数据结构&#xff1f; 堆&#xff08;Heap&#xff09;是一种特殊的完全二叉树&#xff0c;它满足以下核心性质&#xff1a; 堆序性&#xff1a;每个节点的值都满足特定顺序关系 结构性&#xff1a;完全二叉树的结构特性&#xff08;除最后一…

实现悬浮按钮拖动,兼容h5和微信小程序

h5用js写&#xff0c;微信小程序用 代码里面没有完全实现吸附边缘的功能&#xff0c;需要吸附边缘的话还得自己再完善下&#xff08;h5的吸附边缘是可以的&#xff0c;小程序的还有点问题&#xff09; 主要功能是&#xff1a;图片上写文字的悬浮按钮&#xff0c;文字使用的是…

Java程序开发之Spring Security实战:JWT实现登录鉴权

一、JWT与安全认证核心原理 1. JWT结构解析 Header(头部) { "alg": "HS256", "typ": "JWT" } Payload(负载) { "sub": "user123", "exp": 1680403200, "roles": ["US…

嵌入式硬件: GPIO与二极管基础知识详解

1. 前言 在嵌入式系统和硬件开发中&#xff0c;GPIO&#xff08;通用输入输出&#xff09;是至关重要的控制方式&#xff0c;而二极管作为基础电子元件&#xff0c;广泛应用于信号整流、保护电路等。本文将从基础原理出发&#xff0c;深入解析GPIO的输入输出模式&#xff0c;包…

PyTorch 深度学习实战(14):Deep Deterministic Policy Gradient (DDPG) 算法

在上一篇文章中&#xff0c;我们介绍了 Proximal Policy Optimization (PPO) 算法&#xff0c;并使用它解决了 CartPole 问题。本文将深入探讨 Deep Deterministic Policy Gradient (DDPG) 算法&#xff0c;这是一种用于连续动作空间的强化学习算法。我们将使用 PyTorch 实现 D…

【C++】STL全面简介与string类的使用(万字解析)

文章目录 一、STL简介1. 什么是STL2. STL的版本3. STL的组成4. 如何学习STL 二、string类的使用第一部分第二部分第三部分第四部分(含auto与范围for)第五部分第六部分第七部分 一、STL简介 1. 什么是STL STL(standard template libaray-标准模板库)&#xff1a;是C标准库的重要…

[JAVASE] Collection集合的遍历

一. 集合分类 java中的Collection集合分为两类, 分别是单列集合(List)和双列(Map)集合. 1.1 单列集合 1.2 双列集合 二. 集合遍历 2.1 List单列集合的遍历 for each遍历 迭代器遍历 lambda遍历 2.2 Map双列集合的遍历 for each遍历 k-v整体遍历 lambda表达式遍历