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

Java动态代理超详细解析:三步+内存图(堆栈分析)

0.本文适合谁?

1.刚学Java动态代理,被ProxyInvocationHandler绕晕的同学

2.想彻底搞懂代理对象在内存中如何工作的开发者

3.需要面试突击动态代理知识的求职
 

1.动态代理的本质(三步走)

动态代理的核心可以拆解为三个关键步骤:

(1)🔹 第一步:代理类持有目标对象引用

 

public class Person implements InvocationHandler {private Object target;  // ⭐关键!代理类持有目标对象public Person(Object target) {this.target = target;}
}

内存状态:

(2) 🔹 第二步:获取目标类方法

public Object invoke(Object proxy, Method method, Object[] args) {// method就是目标类的核心方法(如zf())return method.invoke(target, args);
}

方法调用栈:
[栈帧]
invoke(proxy, Method(zf), args)
    └── ZF.zf("小白","小黑",100.00)  // 真实调用

(3) 第三步:调用目标方法
 

proxy.zf("小白","小黑",100.00); 
// 实际执行流程:
// 1. 调用$Proxy0.zf() 
// 2. 转发给handler.invoke() 
// 3. 最终调用ZF.zf()

2.完整内存模型 

 🔵 方法调用时(栈堆联动)

proxy.zf("小白","小黑",100.00);

 3.执行流程

  1. 栈帧1main()调用proxy.zf()

  2. 栈帧2$Proxy0.zf()被调用(动态生成的代理方法)

  3. 栈帧3Person.invoke()执行拦截逻辑

  4. 栈帧4:最终调用ZF.zf()

4.完整的代码+运行结果 

 (1)接口定义 IZF.java

/*** 转账接口(抽象主题)*/
public interface IZF {void zf(String name1, String name2, double money);
}

(2)真实实现类 ZF.java

/*** 真实转账类(真实主题)*/
public class ZF implements IZF {@Overridepublic void zf(String name1, String name2, double money) {System.out.println(name1 + "向" + name2 + "转账" + money + "元");}
}

 (3)代理处理器 Person.java

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;/*** 代理处理器(实现InvocationHandler)*/
public class Person implements InvocationHandler {private Object target;  // 持有目标对象引用(关键点1)public Person(Object target) {this.target = target;}/*** 生成代理对象(关键点2)*/public Object getProxy() {return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this  // 传入InvocationHandler);}/*** 方法拦截(关键点3)*/@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("【代理前置操作】记录日志");Object result = method.invoke(target, args);  // 调用真实方法System.out.println("【代理后置操作】更新余额");return result;}
}

 (4)测试类 Test.java

/*** 测试动态代理*/
public class Test {public static void main(String[] args) {// 1. 创建真实对象IZF realObj = new ZF();// 2. 创建代理处理器(传入真实对象)Person handler = new Person(realObj);// 3. 获取代理对象(必须用接口接收!)IZF proxy = (IZF) handler.getProxy();// 4. 通过代理对象调用方法proxy.zf("小黑", "小白", 100.00);}
}

实现结果:
 

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

相关文章:

  • 批量统计PDF页数,统计图像属性
  • dify插件接入fastmcp示例
  • Flink 实时数据一致性与 Exactly-Once 语义保障实战
  • Linux架构篇、第四章_ELK与EFK-7.17.9的日志管理
  • 深入解析Vue3中ref与reactive的区别及源码实现
  • PPT图表怎么制作?说5款自己使用过的PPT图表制作工具
  • 《Python星球日记》 第55天:迁移学习与预训练模型
  • Linux系统管理与编程16:PXE自动化安装部署centos7.9操作系统
  • 计算机设计大赛山东省赛区软件开发赛道线上答辩复盘
  • 基于STM32的甲醛检测
  • 金丝猴食品:智能中枢AI-COP构建全链路数智化运营体系
  • 如何快速分享服务器上的文件
  • SemanticSplitterNodeParser 和 Sentence-BERT 的区别和联系是什么
  • Spring Security 深度解析:打造坚不可摧的用户认证与授权系统
  • 双同步坐标锁相环DDSRF-PLL原理说明
  • 基于 Q-learning 的城市场景无人机三维路径规划算法研究,可以自定义地图,提供完整MATLAB代码
  • Pytorch应用 小记 第一回:基于ResNet网络的图像定位
  • LSP里氏替换原则
  • tmux + ttyd 原理
  • FHE 之 面向小白的引导(Bootstrapping)
  • ISP(Image Signal Processor)处理流程及不同域划分
  • 初等数论--莫比乌斯函数
  • STM32硬件I2C驱动OLED屏幕
  • [文献阅读] wav2vec: Unsupervised Pre-training for Speech Recognition
  • 优选算法——队列+BFS
  • Spark的三种部署模式及其特点与区别
  • GitHub 趋势日报 (2025年05月09日)
  • HTTP:十三.HTTP日志
  • 如何解决 PowerShell 显示 “此系统上禁用了脚本运行” 的问题
  • DAMA语境关系图汇总及考前须知