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

JDK动态代理

首先通俗的介绍什么是代理,我以自身的一个例子来说吧,主播特别喜欢一款游戏,那个游戏做任务会给很多奖励,但是呢我平常没有那么多时间来做任务,又想要得做完任务给的奖励,所以我找了一个游戏带打,可以帮我刷任务,我可以直接想用奖励就行了,美滋滋啦。

下面让我来从代理模式来分析一下吧,上面中的我就是被代理的对象,叫做目标类,而我找的代打叫做代理对象。

动态代理-JDK动态代理

要求目标对象必须要实现至少一个接口,java动态代理会创建实现了相同接口的代理类,然后在运行时动态生成该代理类的实例。

那么JDK动态代理为什么要实现至少一个接口?

因为JDK动态代理的核心类是java.lang.reflect.Prox,这个类的实现方式如下:

其次,JDK在动态代理的时候会生成一个新的类,这个类继承于java.lang.reflect.Prox,由于java是单继承,代理类已经实现了继承proxy,所以只能通过实现接口的方式来代理方法

下面我来讲讲JDK动态代理实现流程

DK动态代理通过Proxy类创建代理对象,所有对代理对象的方法调用都会被转发到实现了InvocationHandler接口的处理器(handler)的invoke()方法中,从而实现方法调用的拦截和增强。

下面让我来分步讲解

首先创建业务类的接口和实现类这俩就不细讲了,直接上车

现在我们已经有了目标类,也就是上图的OrderService类,然后来创建代理类

代理类呢是通过拦截目标类的方法调用,通过代理类来调用目标类的方法,所以呢先通过静态工厂方法创建代理类也就是先找一个帮我刷任务的人

下面让我来解释一下创建代理对象要传的三个参数

ClassLoader loader:动态加载代理类到JVM,确保代理类和目标类在同一个类加载器作用域中

有以下几种方式

// 通常使用目标对象的类加载器 ClassLoader loader = target.getClass().getClassLoader(); // 也可以使用线程上下文类加载器 ClassLoader loader = Thread.currentThread().getContextClassLoader(); // 或使用接口的类加载器 ClassLoader loader = UserService.class.getClassLoader();

Class<?>[] interfaces:定义代理类要实现哪些接口

// 获取目标对象实现的所有接口 Class<?>[] interfaces = target.getClass().getInterfaces(); // 例如返回:[UserService.class, Serializable.class] // 也可以手动指定接口 Class<?>[] interfaces = new Class[]{UserService.class};

如果返回了空数组就会创建失败,这也是为什么目标对象至少要实现一个接口

InvocationHandler :处理器

拦截所有代理对象的方法调用,并且通过重写拦截器中invoke()方法来实现对目标类功能的一些加强,比如添加日志之类的

MyInvocationHandler处理器类实现了InvocationHandler接口

private Object target; 存储真实业务对象的引用(如OrderServiceImpl),通过构造器注入依赖

这儿有一个问题,依赖注入为什么要用构造器注入,set注入不行吗?

1.多线程环境下,有可能没有进行setTarget(),别的线程就进行调用,此时target为null

例如:

2.InvocationHandler要求创建后立即可用

构造器注入在实例化时完成了属性注入,而setter注入是先进行实例化再进行实例化,target可能不完整

3. 使用构造器注入 + final:不可变,线程安全,而setter注入对注入字段不能加final关键字

下面讲一下重写invoke函数的参数意义

proxy-代理对象引用:动态生成的代理类实例,用于方法转发

method-方法元数据:包含被调用方法的所有信息,用于反射调用

args-实际参数值:调用时传入的具体参数值数组

method.invoke():是通过反射机制调用目标对象类,target是目标对象类,args是调用方法时传入的参数。

下面就是测试类了

JDK动态代理主要流程如下:

客户端 → 代理对象 → InvocationHandler.invoke() → 反射调用真实对象方法

总结下来就是一句话:

JDK动态代理=Proxy创建 +InvocationHandler拦截 +接口约束,实现运行时的无侵入式方法增强

http://www.gsyq.cn/news/100279.html

相关文章:

  • TCP53端口和UDP53端口
  • Android企业微信打卡助手使用指南:实现便捷打卡体验
  • 系统与架构,进程,组织,组件,插件,构建之间的逻辑关系和工作机制
  • 离线语音识别新标杆:Whisper.cpp全方位实践指南
  • Zepp Life刷步数完整解决方案:从入门到精通的技术实践
  • Xcode调试救星:iOS设备支持文件快速部署全攻略
  • Springboot居民社区健康管理平台ue77n(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
  • Shutter Encoder终极指南:专业视频编码工具快速上手
  • day39 模型可视化与推理
  • Android弹窗终极指南:BasePopup快速上手与实战技巧
  • Zotero-GPT插件API密钥配置终极指南:3步解决密钥错误问题
  • 国家自然科学基金 申请项目评审意见反馈信
  • 暗黑3智能助手:解放双手的终极自动化解决方案
  • B站4K视频下载完整教程:2025最新工具快速上手指南
  • AI Agent领域的痛点与创新解决方案
  • Screenbox媒体播放器终极方案:Windows生态的完整播放指南
  • 如何用SpiffWorkflow快速构建Python业务流程自动化系统
  • ViGEmBus游戏控制器仿真驱动:轻松解决设备兼容性问题
  • 生成式AI商业化浪潮下的伦理风险与治理之道
  • 25分钟完成700+飞书文档批量导出,这款神器让团队迁移效率提升20倍
  • 告别原生下拉框:Layui formSelects多选插件让表单交互更优雅
  • 视频超分辨率神器:大文件分段处理实战指南
  • 突破百度网盘限速的技术解决方案矩阵
  • 3步搞定Formily集成第三方UI库:从困惑到精通
  • Umi-OCR多引擎适配实战:如何为不同场景选择最优识别方案
  • Formily终极指南:5步搞定第三方UI库无缝集成
  • PIO = Program Input/Output(程序输入输出)
  • Screenbox媒体播放器完整使用指南:Windows平台的终极视频解决方案
  • 终极桌面整理神器:NoFences让Windows工作区焕然一新
  • Java Excel处理性能革命:FastExcel如何实现20倍加速