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

下面先来实现一个动态代码的程序:

先新建一个接口:

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

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

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

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

那用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动态代理运作机制的更多相关文章

  1. JVM插码之四:Java动态代理机制的对比(JDK 和CGLIB,Javassist,ASM)

    一.class文件简介及加载 Java编译器编译好Java文件之后,产生.class 文件在磁盘中.这种class文件是二进制文件,内容是只有JVM虚拟机能够识别的机器码.JVM虚拟机读取字节码文件, ...

  2. ASM字节码框架学习之动态代理

    ASM字节码操纵框架,可以直接以二进制的形式来来修改已经存在的类或者创建新的类.ASM封装了操作字节码的大部分细节,并提供了非常方便的接口来对字节码进行操作.ASM框架是全功能的,使用ASM字节码框架 ...

  3. 从源码角度学习Java动态代理

    前言 最近,看了一下关于RMI(Remote Method Invocation)相关的知识,遇到了一个动态代理的问题,然后就决定探究一下动态代理. 这里先科普一下RMI. RMI 像我们平时写的程序 ...

  4. 深度模拟java动态代理实现机制系类之二

    这次我们要实现的是对任意接口,任意的方法进行特定的代理 这里不一样的只有Proxy类,要实现对所有方法进行代理,那么重点就在于获得接口的所有方法 import java.io.File; import ...

  5. 深度模拟java动态代理实现机制系类之一

    上一篇博客是最基本的动态代理原理的实现,因为其固定了接口,固定了代理方法,以及固定了代理的类型,接下来的博客系类将一步步渐入深度介绍java的动态代理的实现原理 ******************* ...

  6. 深度模拟java动态代理实现机制系类之三

    这里的内容就比较复杂了,要实现的是对任意的接口,对任意指定的方法,以及对任意指定的代理类型进行代理,就更真实的模拟出java虚拟机的动态代理机制 罗列一下这里涉及的类.接口之间的关系,方便大家学习.1 ...

  7. Java 动态代理机制分析及扩展

    Java 动态代理机制分析及扩展,第 1 部分 王 忠平, 软件工程师, IBM 何 平, 软件工程师, IBM 简介: 本文通过分析 Java 动态代理的机制和特点,解读动态代理类的源代码,并且模拟 ...

  8. [转]Java 动态代理机制分析及扩展

    引言 Java 动态代理机制的出现,使得 Java 开发人员不用手工编写代理类,只要简单地指定一组接口及委托类对象,便能动态地获得代理类.代理类会负责将所有的方法调用分派到委托对象上反射执行,在分派执 ...

  9. Java动态代理简单应用

    概念 代理模式是基本的设计模式之一,它是开发者为了提供额外的或不同的操作,而插入的用来代替“实际”对象的对象.这些操作通常涉及与“实际”对象的通信,因此代理通常充当着中间人的角色. Java动态代理比 ...

随机推荐

  1. Work? working!

    0.总要想想为什要做这个(目的是什么,优缺点,必要性,不用行不行,产品定位,如何保证最后的稳定性) 1.如何逐步排查问题所在(做实验+分析排查:不同的实验手段) 2.如何利用网络(搜索问题的所在,确定 ...

  2. 【数据库开发】windows下hiredis的编译(主要是包括一些异步编程的错误)

    果然,高端的程序员真心是鸟都不鸟windows的,Redis的客户端找了一圈愣是没有C++的windows版本 我要做个windows上的C++的服务器都没办法和redis交互 github上所有能试 ...

  3. JS通过ActiveX读写ini配置文件

    String.prototype.trim = function(){ return this.replace(/(^\s+)|(\s+$)/g, ''); }; IniConfig = functi ...

  4. PyCharm Community Edition编辑Python查看数据库

    描述:利用PyCharm Community Edition社区版本的时候无法查看数据库 解决方法如下: 点击File之后选择setting 点击setting之后出现的画面上面选择Plugins,之 ...

  5. SSRAM、SDRAM和Flash简要介绍

    问题1:什么是DRAM.SRAM.SDRAM?答:名词解释如下DRAM--------动态随即存取器,需要不断的刷新,才能保存数据,而且是行列地址复用的,许多都有页模式SRAM--------静态的随 ...

  6. 报错Unsupported major.minor version 52.0的原因

    原因分析 很明显是JDK版本不一致所导致的,,我们知道Java是支持向后编译的,也就是说,在低版本(如JDK6)上编译的class文件或者jar文件可以在高版本上(如JDK8)上编译通过,但这并不意味 ...

  7. [转帖]linux常用命令大全(linux基础命令入门到精通+实例讲解+持续更新+命令备忘录+面试复习)

    linux常用命令大全(linux基础命令入门到精通+实例讲解+持续更新+命令备忘录+面试复习) https://www.cnblogs.com/caozy/p/9261224.html 总结的挺好的 ...

  8. WCF-复杂配置

    两种模式,一个契约两个实现,两个契约一个实现. 服务类库 宿主 static void Main(string[] args) { ServiceHost sh1 = new ServiceHost( ...

  9. Elasticsearch集群搭建笔记(elasticsearch-6.3.0)

    # 检查Java版本 java -version # 安装Elasticsearch,所有节点均安装并解压 wget https://artifacts.elastic.co/downloads/el ...

  10. 《Redis - 穿透/击穿/雪崩/集中失效》

    一:什么是缓存穿透? - 定义 - 正常情况下,我们在理想的条件下去查询缓存数据都是存在的. - 那么请求去查询一条数据库中不存在的数据,也就是缓存和数据库都查询不到这条数据. - 所以请求每次都会打 ...