当前位置: 首页 > ai >正文

Java基础之静态代理和动态代理

一、静态代理

定义

代理类与被代理类在编译期就已确定,代理类需要手动编写并实现与被代理类相同的接口

1.核心角色

  1. Subject(抽象主题):定义业务方法的接口

  2. RealSubject(真实主题):实现接口的具体业务类

  3. Proxy(代理类):实现相同接口,持有真实主题的引用

2.为什么需要静态代理?

        在已经完成的代码中,我们需要对完成的业务进行增强。这时我们又不想改动代码,已经写好的业务代码所以这是我们需要静态代理。

3.代码:

package 静态代理;public interface UserService {public void FindName();
}

package 静态代理;public class UserServiceimpl implements UserService {@Overridepublic void FindName() {System.out.println("正在查找请稍等");}}

package 静态代理;public class Proxyclass implements UserService {private UserServiceimpl userserviceimpl=new UserServiceimpl();@Overridepublic void FindName() {System.out.println("支付完成");userserviceimpl.FindName();}}

package 静态代理;public class TestApp {public static void main(String[] args) {UserService userServie1=new UserServiceimpl();userServie1.FindName();System.out.println("*************************************");UserService userServie2=new Proxyclass();userServie2.FindName();}}

 4.执行流程图

 5.运行结果:

 

二、动态代理

在面向对象开发中,动态代理是一种强大的技术手段,它允许我们在运行时动态创建代理对象,实现对目标方法的增强和拦截。与静态代理相比,动态代理具有以下显著优势:

  1. 代码解耦:将核心业务逻辑与横切关注点(如日志、事务、权限控制)分离

  2. 灵活扩展:无需为每个目标类编写代理类,减少重复代码

  3. 运行时决策:代理逻辑可以根据运行时条件动态调整

1. 核心组件

  • java.lang.reflect.Proxy:负责生成代理类

  • java.lang.reflect.InvocationHandler:方法调用处理器接口

        

2. 实现步骤图解

[客户端] -> [代理对象] -> [InvocationHandler] -> [目标对象]

3. 代码实现解析

package jdk动态代理;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;public class UserServiceimpl_Proxy_Dynamic implements InvocationHandler {private Object object;public Object get (Object o ) {this.object=o;return Proxy.newProxyInstance(this.object.getClass().getClassLoader(), this.object.getClass().getInterfaces(), this);}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {////对象proxy一般用不到,method是被代理对象的方法,args参数System.out.println("动态代理增强了方法");return method.invoke(this.object, args);//前面是外界获取的方法,args参数}
}
package 静态代理;import jdk动态代理.UserServiceimpl_Proxy_Dynamic;public class TestApp {public static void main(String[] args) {UserService userServie1=new UserServiceimpl();userServie1.FindName();System.out.println("*************************************");UserService userServie2=new Proxyclass();userServie2.FindName();System.out.println("下面是动态代理方法*****************************");UserServiceimpl_Proxy_Dynamic userServiceimpl_proxy_dynamic=new UserServiceimpl_Proxy_Dynamic();UserService userService=(UserService) userServiceimpl_proxy_dynamic.get(userServie1);userService.FindName();}}

4. 关键参数说明

参数作用说明
ClassLoader定义代理类的加载器,通常使用目标类的类加载器
Class<?>[] interfaces代理类要实现的接口列表,决定代理对象支持哪些方法
InvocationHandler方法调用处理器,所有代理方法调用都会路由到该接口的invoke方法

  5.运行结果:

6.代理方案对比 

特性静态代理JDK动态代理CGLIB代理
实现方式手动编写代理类接口代理子类继承
编译期处理需要不需要不需要
执行效率中等(反射调用)较高(ASM生成字节码)
目标类要求无特殊要求必须实现接口不能是final类
方法拦截范围显式实现的接口方法接口方法所有非final方法

7.实践建议 

  1. 优先选择JDK动态代理:当目标类已实现接口时

  2. 合理使用缓存:对重复创建的代理对象进行缓存

  3. 异常处理:在invoke方法中妥善处理异常

  4. 组合使用:结合模板方法模式实现通用代理逻辑

  5. 性能监控:在关键路径上记录代理耗时

http://www.xdnf.cn/news/6068.html

相关文章:

  • ValueError: 4 columns passed, passed data had 51141 columns解决
  • 【消息队列】RabbitMQ基本认识
  • Git仓库迁移
  • 深度解析 Sora:从技术原理到多场景实战的 AI 视频生成指南【附学习资料包下载】
  • 模糊数学方法之模糊贴近度
  • 现代 Web 自动化测试框架对比:Playwright 与 Selenium 的深度剖析
  • AI智能分析网关V4周界入侵检测算法精准监测与智能分析,筑牢周界安全防线
  • flutter 视频通话flutter_webrtc
  • @Controller 与 @RestController-笔记
  • 架构设计不合理,如何优化系统结构
  • 设计并实现高并发系统,应用无锁编程与CAS机制
  • Android usb网络共享详解
  • Linux笔记---信号(中)
  • 计算机视觉----基础概念、卷积
  • 基于javaweb的SpringBoot自习室预约系统设计与实现(源码+文档+部署讲解)
  • VUE3 -综合实践(Mock+Axios+ElementPlus)
  • 基于Matlab的非线性Newmark法用于计算结构动力响应
  • 如何查看打开的 git bash 窗口是否是管理员权限打开
  • Oracle 中的虚拟列Virtual Columns和PostgreSQL Generated Columns生成列
  • win11 安装 wsl ubuntu 18.04后换源失败!
  • Void: Cursor 的开源平替
  • ET MessageQueue类分析
  • 汽车免拆诊断案例 | 2015款路虎极光车组合仪表提示“充电系统故障”
  • 第二个五年计划!
  • Android清单文件
  • No module named ‘OpenGL‘
  • 【SSL部署与优化​】​​HTTP/2与HTTPS的协同效应
  • Python uv包管理器使用指南:从入门到精通
  • 5.14本日总结
  • 地磁-惯性-视觉融合制导系统设计:现代空战导航的抗干扰解决方案