当前位置: 首页 > 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);}
}

实现结果:
 

相关文章:

  • Linux进程间信号
  • ts装饰器
  • 从杰夫・托尔纳看 BPLG 公司的技术创新与发展
  • LeetCode 39 LeetCode 40 组合总和问题详解:回溯算法与剪枝优化(Java实现)
  • Python爬虫实战:获取woodo网各类免费图片,积累设计素材
  • [题解]2023CCPC黑龙江省赛 - Folder
  • 服务预热原理
  • 批量统计PDF页数,统计图像属性
  • 求数组中的两数之和--暴力/哈希表
  • Java 23种设计模式 - 行为型模式11种
  • JAVA——抽象类和接口的区别
  • A2A大模型协议及Java示例
  • jdk多版本切换,通过 maven 指定编译jdk版本不生效,解决思路
  • 使用lldb查看Rust不同类型的结构
  • cv_connection (像halcon一样对区域进行打散)
  • Markdown—LaTeX 数学公式
  • PCB设计实践(十二)PCB设计电容选型:功能、材质、规则
  • 数据结构与算法分析实验12 实现二叉查找树
  • 问题及解决01-面板无法随着窗口的放大而放大
  • 【论文阅读】Efficient and secure federated learning against backdoor attacks
  • 呼和浩特推进新一轮国企重组整合:杜绝一项目一公司、一业务一公司
  • 丹麦召见美外交官,强调“不能容忍”美在格陵兰岛间谍活动
  • 甘肃省政府原副省长赵金云被决定逮捕
  • 商务部:中方愿同各国一道加强合作,促进跨境电商健康可持续发展
  • 对话哭泣照被恶意盗用成“高潮针”配图女生:难过又屈辱
  • 上海优化营商环境十大攻坚突破任务中,为何第一项是实施世行对标改革?