透过字节码生成审视Java动态代理运作机制
对于动态代理我想应该大家都不陌生,就是可以动态去代理实现某个接口的类来干一些我们自己想要的功能,但是在字节码层面它的表现是如何的呢?既然目前刚好在研究字节码相关的东东,有必要对其从字节码角度来审视一下。
下面先来实现一个动态代码的程序:
先新建一个接口:

接下来定义一个具体的实现类:

然后再建议一个动态代理类,如下:

然后在里面要持有要代理的真实对象,如下:

然后咱们来调用一下看下效果:

那用Proxy生成代理对象都传些啥呢,第一个参数是加载动态对象的classloader,这里传加载RealSubject类的类加载器,如下:

第二个参数为生成动态对象所实现的接口,这里直接拿真实对象的接口既可:

最后一个参数则传InvocationHandler对象,如下:

接下来运行一下:

另外注意一下,其中动态生成的subject接口的对像真实类型其实是一个代理类,咱们可以打印一下:

其中“com.sun.proxy.$Proxy0”在是运行期动态生成的代理对像,而不是我们自己编定的类,那该对象对应的字节码信息要怎么看呢?所以第一个难题就出来了,要想知道它的字节码信息,就必须知道它是如何生成的,所以别无它法,只能硬着头皮去分析源代码才能知晓,所以接下来去源码来寻找蛛丝马迹:
根据调用咱们就必须跟到Proxy中的newProxyInstance()方法去查看喽,走进去!

其中先对这三个参数再来瞅一下:



接下来开始寻找线索,注意:只看跟我们预期相符的主要代码,一些无关的直接略过,发现有一处貌似挺关键的,如下:

所以跟进去:


既然上面的这个关键注释可以看刚我们所要查找的代理类的对像很显然跟这个缓存有关,所以跟进去:

但是我们会发现这个方法中貌似没有太多线索,这里就不贴出该方法的所有源码了, 不过,我们可以将焦点先定位到方法结果的返回上,也就是这:

所以接下来就得知道supplier对像是如何创建出来的,所以可以往下找就会发现有这样一段代码:

然后咱们就进入这个Factory去瞅一眼:

那。。没有办法的时候可以用debug的方式来走一遍流程,所以下面来debug一下:

然后往里跟,因为代理对像的生成就是在这个方法里面:

继续往里跟:

然后在这里面单步调试时会到我们之前分析的Factory生成这块:

此时则生成了一个factory对象,接着往下到关键部位:

此时将factory赋值给了supplier对象了,接着往下又会回到循环开始处:

此时再往下单步走一步,就会发现我们的预期要的东东啦,如下:

所以整个分析的焦点还得放在supplier.get()方法来,而它是由Factory赋值的,所以又回到了之初我们分析的Factory对像啦,所以咱们在Factory里面的get()方法打个断点继续来找寻:


所以跟进去看一下:


此时又回到了Proxy代理类了,咱们往下单步跟踪到关键处如下:

线索最终找到,也就是最终代理类的生成是ProxyGenerator这个类来实现的,最终得到一个二进制的字节码对象,咱们跟进去瞅一下这个类:

然后定位到指定的方法就可以发生如何来生成代理类的字节文件出来了,如下:

所以现在就是想办法将这个flag设置为true既可,那看一下这个flag:

所以接下来咱们来设置一下该系统属性看下效果:

咱们来运行一下:

下次就来分析该字节码!
透过字节码生成审视Java动态代理运作机制的更多相关文章
- JVM插码之四:Java动态代理机制的对比(JDK 和CGLIB,Javassist,ASM)
一.class文件简介及加载 Java编译器编译好Java文件之后,产生.class 文件在磁盘中.这种class文件是二进制文件,内容是只有JVM虚拟机能够识别的机器码.JVM虚拟机读取字节码文件, ...
- ASM字节码框架学习之动态代理
ASM字节码操纵框架,可以直接以二进制的形式来来修改已经存在的类或者创建新的类.ASM封装了操作字节码的大部分细节,并提供了非常方便的接口来对字节码进行操作.ASM框架是全功能的,使用ASM字节码框架 ...
- 从源码角度学习Java动态代理
前言 最近,看了一下关于RMI(Remote Method Invocation)相关的知识,遇到了一个动态代理的问题,然后就决定探究一下动态代理. 这里先科普一下RMI. RMI 像我们平时写的程序 ...
- 深度模拟java动态代理实现机制系类之二
这次我们要实现的是对任意接口,任意的方法进行特定的代理 这里不一样的只有Proxy类,要实现对所有方法进行代理,那么重点就在于获得接口的所有方法 import java.io.File; import ...
- 深度模拟java动态代理实现机制系类之一
上一篇博客是最基本的动态代理原理的实现,因为其固定了接口,固定了代理方法,以及固定了代理的类型,接下来的博客系类将一步步渐入深度介绍java的动态代理的实现原理 ******************* ...
- 深度模拟java动态代理实现机制系类之三
这里的内容就比较复杂了,要实现的是对任意的接口,对任意指定的方法,以及对任意指定的代理类型进行代理,就更真实的模拟出java虚拟机的动态代理机制 罗列一下这里涉及的类.接口之间的关系,方便大家学习.1 ...
- Java 动态代理机制分析及扩展
Java 动态代理机制分析及扩展,第 1 部分 王 忠平, 软件工程师, IBM 何 平, 软件工程师, IBM 简介: 本文通过分析 Java 动态代理的机制和特点,解读动态代理类的源代码,并且模拟 ...
- [转]Java 动态代理机制分析及扩展
引言 Java 动态代理机制的出现,使得 Java 开发人员不用手工编写代理类,只要简单地指定一组接口及委托类对象,便能动态地获得代理类.代理类会负责将所有的方法调用分派到委托对象上反射执行,在分派执 ...
- Java动态代理简单应用
概念 代理模式是基本的设计模式之一,它是开发者为了提供额外的或不同的操作,而插入的用来代替“实际”对象的对象.这些操作通常涉及与“实际”对象的通信,因此代理通常充当着中间人的角色. Java动态代理比 ...
随机推荐
- 三节课MINI计划第四周
一.任务 二.任务一 (一)梳理产品卖点 (二)策划线上活动 三.任务二 四.周报
- .git泄露及利用php弱类型松散比较构造json的payload
一道ctf题,文章搬运到了自己的网站上: http://101.132.137.140:202/archives/2019-11-16
- 记:联调安卓设备的神药-无需usb数据线即可直连
前言 最近需要调试公司的安卓服务,正常情况下,我们调试都是减安卓设备通过usb连接在我们座位旁,再不济就是我们扛笔记本到硬件旁边,这样调试屡试不爽,但是有一天你突然发现你带的数据线因为各种原因总是终端 ...
- PTA 8-1 jmu-java-流、文件与正则表达式 (5 分)
0.字节流与文件 我的代码: public static byte[] readFile(String path){ File file = new File(path); FileInputStre ...
- Shuffle 机制
1. 概述 Map 方法之后,Reduce 方法之前的数据处理过程称之为 Shuffle. 2. Partition 分区 需求:要求将统计结果按照条件输出到不同文件中(分区).比如:将统计结果按照手 ...
- 27.Spark中transformation的介绍
Spark支持两种RDD操作:transformation和action.transformation操作会针对已有的RDD创建一个新的RDD: 而action则主要是对RDD进行最后的操作,比如遍历 ...
- AirFlow后台运行调度程序
nohup airflow worker>>$AIRFLOW_HOME/airflow-worker.log >& & nohup airflow scheduler ...
- python 脚本定时删除 elk索引
脚本如下 一.python 脚本如下 #! /usr/bin/python # -*- coding=utf-8 -*- import urllib import urllib.request imp ...
- 数组、可变参数 、this关键字 (札记)
Thinking in java 读书笔记(P84 ~ P104) 作者:淮左白衣 写于:2018年4月10日16:42:57 目录 this 为什么可以代表调用对象 数组 数组中的 length 定 ...
- python 手机App数据抓取实战一
前言 当前手机使用成为互联网主流,每天手机App产生大量数据,学习爬虫的人也不能只会爬取网页数据,我们需要学习如何从手机 APP 中获取数据,本文就以豆果美食为例,讲诉爬取手机App的流程 环境准备 ...