理解加载字节码到JVM的时机
发布于2022-07-22 00:17:50,更新于2023-09-04 20:16:05,标签:java jvm 文章会持续修订,转载请注明来源地址:https://meethigher.top/blog最近有看《深入理解Java虚拟机》,作者很聪明,这边直接一笔带过,跟没提一样。
甚至百度都搜不到,领域大佬直接给大众树了死标杆,由此,我自己来记录踩坑了。
纸上得来终觉浅,绝知此事要躬行,不知道说了多少次。
一、idea远程断点
1.1 两种方式
为啥要远程断点?
我特意干掉了某个依赖,本地可以启动,打包后有bug,我还特意确定了本地也是没有引入的。我怀疑是idea偷偷给我引入了。所以要排查问题,只能远程断点排查了。
远程调试的要点是,本地的代码与远程的jar包是一致的,这样才能保证断点的行数能够对的上。
不管远程的debug还是本地的debug,都是通过远程连接到jvm执行的。至于为啥,我也不知道。
可以搜索相关的JPDA技术,JPDA 的全称是 Java Platform Debugger Architecture,即java平台的调试技术。
准备环境
- 本机机器192.168.101.2
- 本地调试工具idea2021
- 远程机器10.0.0.10
两种debugger模式
- Attach to remote JVM:远程项目先启动,本地主动连接远程已启动好的项目。可能会存在,你连接上时,你的断点已经过了。
- Listen to remote JVM:本地监听先启动,监听到远程项目启动后自动连接。推荐使用这种方式,这也是我们idea直接debug所使用的方式。
下面是具体的两种模式
1.1.1 连接远程服务
连接远程服务对应的模式叫做Attach to remote JVM,该模式需要远程额外开启一个端口,适用于无公网ip的客户端调试公网服务器。
将本机192.168.101.2连接到10.0.0.10的80端口
将生成的命令复制下来,图中所开放端口,默认只监听访问127.0.0.1的流量。
如果想要监听所有本机ip的流量,需修改如下
0.0.0.0含义是未知ip,代表的就是本机的所有ip
1 | -agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=0.0.0.0:80 |
将修改后的参数,合并成服务器启动服务的命令
1 | java -jar -agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=0.0.0.0:80 qiandao-1.0.jar |
1.1.2 远程服务监听
监听远程服务对应的模式叫做Listen to remote JVM,该模式需要客户端额外开启一个端口,让服务端连入。适用于机器互通的情况下使用。
这个模式在我选中Listen to remote JVM之后,IDEA就卡死了,经过各种排查,跟插件等无关。跟win11预览版镜像也无关。
还是跟之前网页解析慢的原因一样的,我有多个网卡,对应有多个ip地址,他卡在找ip地址的环节了。
如果卡住,解决办法是,把wifi关掉,或者去网络共享中心把适配器禁用。
将图中生成的args复制下来,进行相应的修改。
1 | -agentlib:jdwp=transport=dt_socket,server=n,address=192.168.101.2:80,suspend=y |
本地debugger模式启动,远程执行命令启动。
1 | java -jar -agentlib:jdwp=transport=dt_socket,server=n,address=192.168.101.2:80,suspend=y qiandao-1.0.jar |
结果如图
1.2 条件断点
idea在断点处进行右键,输入条件即可,如图。
更多调试技巧,参考IDEA调试技巧
二、通过字节码文件创建对象
加载class字节码分为三个步骤
- loadClass
- findClass
- defineClass
创建一个自定义的类加载器,用于加载One.class字节码。
One.class具体获取在第三节
1 | /** |
加载效果如图
三、理解加载字节码到JVM的时机
其实到这里才是正文。
3.1 应用场景
起因,我写了一套监控数据库的基础组件,基础组件嘛,只提供最基础的功能逻辑,而不提供依赖jar包。等到有人使用我的组件时,按需引入依赖就行了。轻量可复用,多好!
打包时不将依赖打入jar包,只需maven依赖scope改为provided即可。scope默认的是compiled。
我的监控对象如下。
依赖
1 | <dependency> |
具体源码如下
1 | import com.mongodb.client.MongoClient; |
我想实现的效果是,即使我没有使用的那些依赖,仍然能new One()成功。等到实际调用mongoInfo()时,如果没有依赖再报错嘛。
但是,我new One()时,就已经触发了Bson的class加载到JVM。
为了更能直观的查看结果,我直接加载One的字节码。
启动命令带上参数-verbose:class表示查看类加载明细。
为啥要加载字节码?因为用IDEA启动,在没有依赖时,会有编译检查,没法直接运行。
我想模拟没有依赖的环境,两种办法都可行,一种是远程连接jar包,一种是加载字节码。
当然也可以配置idea跳过编译检查,麻烦。
到这里,我就好奇了,为啥Bson还没调用的时候,就load到jvm了?而mongo就不会这样?
为了验证上面的问题,我又加入Bson依赖,再次查看类加载结果,确实没有加载mongo。
这边查询《深入理解Java虚拟机》,作者很聪明,只讲了类初始化时机,类加载的触发条件,闭口不讲。
3.2 bug明晰
先明确几个概念。
类加载:字节码load到jvm
类初始化:碰到new、反射invoke、直接调用类的静态变量or方法时等等,会触发类的初始化。一定程度上这也可以理解成类加载的触发条件,因为类加载后才会初始化嘛。
对象实例化:new Object(),这就表示了实例化了一个类型为Object的对象。
顺序:类加载 -> 类初始化 -> 对象实例化
实例化上述代码中的One对象时,由于One中引用了Document以及MongoClient等。
其中Document实现自Bson接口,MongoClient实现自Closeable接口。
通过启动时,查看类加载过程,可知,虽然只是引用、尚未调用,顶级父类仍然会加载到jvm,即进行类加载。
通过注释掉其中一行涉及到多态的写法,就不会触发Bson的类加载了。
总结来说,类A被调用时,类A引用到的类B直接涉及的父类/父接口(比如多态写法)会立即执行类加载。而引用的类B本身,通常(父级为抽象类时为特殊情况)在被调用时,才会触发其类加载。所以这个问题,只要使用非多态写法即可解决。
非多态写法的类,是需要调用,才会触发加载。
多态写法的,引用到的抽象类和接口的加载情况略有不同。
经测试,父级只要引用到,即使没有调用也会立即加载。对于父级是抽象类的,还会带着子类一同加载。对于父级是接口的,子类只会在被调用时加载。
3.3 思想验证
已经明确的是,非多态写法只有使用到时才会触发类加载。
至于父级抽象类和接口的加载时机,需要测试。
One继承自抽象类AbstractClass,Two实现自InterfaceClass。
1 | public class Three { |
如上案例项目启动,没有引用到父级方法时,jvm只会加载Three。
但是如果引用到父级的方法。
1 | public class Three { |
可以看到引用到父级方法时,
如果父级是接口,则jvm只会加载接口。
如果父级是抽象类,则jvm会加载抽象类和实现类。
3.4 解决方案
我所说的解决方案是不加依赖的情况下。
第一,非多态写法。这个不可行,毕竟这是别人包里的api。
第二,将类的引用,放到另外一个类,利用类本身就是懒加载的特性,实现延迟加载。该思路借鉴单例模式的延迟初始化占位类思想。
代码如下
1 | import com.mongodb.client.MongoClient; |
类的懒加载:类被调用时触发类加载,用到才会加载。
单例的懒加载:类加载完成后,等到初次获取实例时,进行对象的实例化。
单例的急加载:类加载并执行初始化时就立即进行对象的实例化。
3.5 有趣案例
看代码
1 | public class Base { |
运行结果是
1 | Base静态代码块执行 |
通过多态形式实例化Sub,加载顺序如下。
- 父类的静态代码块和静态变量
- 子类的静态代码块和静态变量
- 父类的非静态代码块和非静态变量
- 父类的构造函数
- 子类的非静态代码块和非静态变量
- 子类的构造函数
子类Sub没有构造函数,所以就是默认的public new Sub(){}
由于子类重写了父类的callName方法,所以执行父类构造函数里面callName实际是获取子类的baseName,而此时子类的变量还没有进行加载,所以为null
四、参考致谢
idea,使用Remote 连接tomcat,远程DEBUG模式调试_可乐cc呀的博客-CSDN博客_idea tomcat 远程debug
IDEA远程断点调试jar包项目_单手入天象的博客-CSDN博客_idea 断点进入jar包
基于Java动态编译实现springboot项目动态加载class文件的一些经历和思考_追风小勺年的博客-CSDN博客_springboot 动态加载类
Java多态时类的加载顺序_奋起直追CDS的博客-CSDN博客
spring-boot-starter-undertow和tomcat的区别_芭比萌妹的博客-CSDN博客_undertow和tomcat的区别