本文持久维护地址

最近有看《深入理解Java虚拟机》,作者很聪明,这边直接一笔带过,跟没提一样。

甚至百度都搜不到,领域大佬直接给大众树了死标杆,由此,我自己来记录踩坑了。

纸上得来终觉浅,绝知此事要躬行,不知道说了多少次。

一、idea远程断点

1.1 远程断点基本步骤

为啥要远程断点?

我特意干掉了某个依赖,本地可以启动,打包后有bug,我还特意确定了本地也是没有引入的。我怀疑是idea偷偷给我引入了。所以要排查问题,只能远程断点排查了。

远程调试的要点是,本地的代码与远程的jar包是一致的,这样才能保证断点的行数能够对的上。

不管远程的debug还是本地的debug,都是通过远程连接到jvm执行的。至于为啥,我也不知道。

可以搜索相关的JPDA技术,JPDA 的全称是 Java Platform Debugger Architecture,即java平台的调试技术。

准备环境

  • 本机机器192.168.110.199
  • 本地调试工具idea2021
  • 远程机器10.0.0.10

两种debugger模式

  1. Attach to remote JVM:远程项目先启动,本地主动连接远程已启动好的项目。可能会存在,你连接上时,你的断点已经过了。
  2. Listen to remote JVM:本地监听先启动,监听到远程项目启动后自动连接。推荐使用这种方式,这也是我们idea直接debug所使用的方式。

将图中生成的args复制下来,进行相应的修改。

-agentlib:jdwp=transport=dt_socket,server=n,address=192.168.110.199:54188

本地debugger模式启动,远程执行命令启动。

java -agentlib:jdwp=transport=dt_socket,server=n,address=192.168.110.199:54188 -jar class-load-1.0.0.jar

结果如图

1.2 条件断点

idea在断点处进行右键,输入条件即可,如图。

二、通过字节码文件创建对象

创建一个自定义的类加载器,用于加载One.class字节码。

One.class具体获取在第三节

/*** 自定义类加载器** @author chenchuancheng github.com/meethigher* @since 2022/7/22 21:58*/
public class CustomClassLoader extends ClassLoader {private final byte[] bytes;public CustomClassLoader(String classPath) {try {FileInputStream fis = new FileInputStream(classPath);ByteArrayOutputStream bos = new ByteArrayOutputStream();int b;while ((b = fis.read()) != -1) {bos.write(b);}this.bytes = bos.toByteArray();bos.close();fis.close();} catch (Exception e) {throw new RuntimeException(e.getMessage());}}@Overridepublic Class<?> loadClass(String name) throws ClassNotFoundException {// 首先,检查class是否已经被加载Class<?> c = findLoadedClass(name);// 类名不是指定的类的,默认使用委托的父类加载器if (!name.equals("One")) {ClassLoader parent = getParent();c = parent.loadClass(name);}//如果为空,就将字节数组转换为类if (c == null) {c = defineClass(name, bytes, 0, bytes.length);}return c;}
}

加载效果如图

三、理解加载字节码到JVM的时机

其实到这里才是正文。

3.1 应用场景

起因,我写了一套监控数据库的基础组件,基础组件嘛,只提供最基础的功能逻辑,而不提供依赖jar包。等到有人使用我的组件时,按需引入依赖就行了。轻量可复用,多好!

打包时不将依赖打入jar包,只需maven依赖scope改为provided即可。scope默认的是compiled。

我的监控对象如下。

依赖

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-mongodb</artifactId><version>2.5.2</version><exclusions><exclusion><groupId>org.mongodb</groupId><artifactId>bson</artifactId></exclusion></exclusions>
</dependency>

具体源码如下

import com.mongodb.client.MongoClient;
import com.mongodb.client.MongoClients;
import com.mongodb.client.MongoDatabase;
import com.mongodb.client.MongoIterable;
import org.bson.Document;public class One {public One() {}public void OnePrint() {System.out.println("哈哈哈哈");}public void mongoInfo() {MongoClient client = MongoClients.create();MongoDatabase db = client.getDatabase("test");MongoIterable<String> collectionNames = db.listCollectionNames();String str = String.format("{collStats : '%s'}", "test");//B为单位Document document = Document.parse(str);document = db.runCommand(document);System.out.println(document);}
}

我想实现的效果是,即使我没有使用的那些依赖,仍然能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。

public class Three {public void test() {InterfaceClass aClass = new Two();}public void test1() {AbstractClass aClass = new One();}public static void main(String[] args) {new Three();}
}

如上案例项目启动,没有引用到父级方法时,jvm只会加载Three。

但是如果引用到父级的方法。

public class Three {public void test() {InterfaceClass aClass = new Two();aClass.test();}public void test1() {AbstractClass aClass = new One();aClass.test();}public static void main(String[] args) {new Three();}
}

可以看到引用到父级方法时,

如果父级是接口,则jvm只会加载接口。

如果父级是抽象类,则jvm会加载抽象类和实现类。

3.4 解决方案

我所说的解决方案是不加依赖的情况下。

第一,非多态写法。这个不可行,毕竟这是别人包里的api。

第二,**将类的引用,放到另外一个类,利用类本身就是懒加载的特性,实现延迟加载。**该思路借鉴单例模式的延迟初始化占位类思想。

代码如下

import com.mongodb.client.MongoClient;
import com.mongodb.client.MongoClients;
import com.mongodb.client.MongoDatabase;
import com.mongodb.client.MongoIterable;
import org.bson.Document;public class One {public One() {}public void OnePrint() {System.out.println("哈哈哈哈");}public void mongoInfo() {MongoClient client = MongoClients.create();MongoDatabase db = client.getDatabase("test");MongoIterable<String> collectionNames = db.listCollectionNames();String str = String.format("{collStats : '%s'}", "test");//B为单位System.out.println(OneHolder.hold(db, str));}private static final class OneHolder {/*** 这样就相当于将引用放到了OneHolder这个类这里* 只有OneHolder被调用时,才会触发引用到的类的父级的加载** @param db* @param str* @return*/private static Document hold(MongoDatabase db, String str) {Document document = Document.parse(str);return db.runCommand(document);}}public static void main(String[] args) {new One();}
}

类的懒加载:类被调用时触发类加载,用到才会加载。

单例的懒加载:类加载完成后,等到初次获取实例时,进行对象的实例化。

单例的急加载:类加载并执行初始化时就立即进行对象的实例化。

3.5 有趣案例

看代码

public class Base {private String baseName = "base";static {System.out.println("Base静态代码块执行");}public Base() {System.out.println("Base构造函数执行");callName();}public void callName() {System.out.println(baseName);}static class Sub extends Base {private String baseName = "sub";static {System.out.println("Sub静态代码块执行");}public Sub() {System.out.println("Sub构造函数执行");}public void callName() {System.out.println(baseName);}}public static void main(String[] args) {Base b = new Sub();//null}
}

运行结果是

Base静态代码块执行
Sub静态代码块执行
Base非静态代码块执行
Base构造函数执行
null
Sub非静态代码块执行
Sub构造函数执行

通过多态形式实例化Sub,加载顺序如下。

  1. 父类的静态代码块和静态变量
  2. 子类的静态代码块和静态变量
  3. 父类的非静态代码块和非静态变量
  4. 父类的构造函数
  5. 子类的非静态代码块和非静态变量
  6. 子类的构造函数

子类Sub没有构造函数,所以就是默认的public new Sub(){}

由于子类重写了父类的callName方法,所以执行父类构造函数里面callName实际是获取子类的baseName,而此时子类的变量还没有进行加载,所以为null

四、参考致谢

idea,使用Remote 连接tomcat,远程DEBUG模式调试_可乐cc呀的博客-CSDN博客_idea tomcat 远程debug

IDEA远程断点调试jar包项目_单手入天象的博客-CSDN博客_idea 断点进入jar包

怎样在IDEA中设置条件断点-百度经验

用.class文件创建对象 - 冷冰鱼 - 博客园

基于Java动态编译实现springboot项目动态加载class文件的一些经历和思考_追风小勺年的博客-CSDN博客_springboot 动态加载类

Java多态时类的加载顺序_奋起直追CDS的博客-CSDN博客

spring-boot-starter-undertow和tomcat的区别_芭比萌妹的博客-CSDN博客_undertow和tomcat的区别

理解加载class到JVM的时机相关推荐

  1. Unity 全面理解加载和内存管理

    最近一直在和这些内容纠缠,把心得和大家共享一下: Unity里有两种动态加载机制:一是Resources.Load,一是通过AssetBundle,其实两者本质上我理解没有什么区别.Resources ...

  2. java 数据加载到内存jvm中

    为什么需要将java 数据加载到内存? 1 将数据加载到jvm运行内存中,会占用运行内存,一些对象,初始化数据,枚举等 缺点:如果值有修改,需要重新部署项目才能生效. 2 一些不想放到redis 缓存 ...

  3. 对HashMap数据结构的理解——加载因子和初始容量

    先看源码: 解释一下位移运算: 1<<4 是位移运算的表示,为十进制16 1的二进制表示:1 左移4位之后的二进制表示为B(10000) = D(16) 更简单的计算方法就是 1<& ...

  4. jvm系列(一):java类的加载机制

    1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个 java.lang.Class对象,用来封装类在方法区内的数据结 ...

  5. 这篇文章绝对让你深刻理解java类的加载以及ClassLoader源码分析

    前言 package com.jvm.classloader;class Father2{public static String strFather="HelloJVM_Father&qu ...

  6. JVM篇2:[-加载器ClassLoader-]

    写本篇的动因只是一段看起来很诡异的代码,让我感觉有必要认识一下ClassLoader ----[Counter.java]------------------------- public class ...

  7. JVM中篇:字节码与类的加载篇

    0.概述 0.1字节码文件的跨平台性 0.1.1.Java语言:跨平台的语言(write once,run anywhere) 当]ava源代码成功编译成字节码后,如果想在不同的平台上面运行,则无须再 ...

  8. jvm加载class原理

    转载地址 : http://hxraid.iteye.com/blog/747625 当Java编译器编译好.class文件之后,我们需要使用JVM来运行这个class文件.那么最开始的工作就是要把字 ...

  9. 深入理解Java虚拟机:Java类的加载机制

    本篇内容包括:Java 类的加载机制(Jvm 结构组成.Java 类的加载).类的生命周期(加载-验证-准备-解析-初始化-使用-卸载).类加载器 以及 双亲委派模型. 一.Java 类的加载机制 1 ...

  10. 从JVM看类的加载过程与对象实例化过程

    一. 类的加载过程 1. 类的加载过程大致是个什么过程? 我们编写产生.java文件,这些.java文件经过Java编译器编译成拓展名为.class的文件,.class文件中保存着Java代码经转换后 ...

最新文章

  1. spark+数据倾斜+解决方案
  2. javascript中的事件问题的总结
  3. oracle 表连接 大表小表_优化必备基础:Oracle中常见的三种表连接方式
  4. java k线绘制,用Java绘制K线图[Java编程]
  5. 借助传感器用计算机测速度实验题,专家分析2015年高考命题趋势 内容设计将再创新...
  6. paho mqtt 订阅主题的处理注意事项
  7. Java连接open fire_java – 为什么我不能连接到openfire服务器?
  8. L3-020 至多删三个字符 [DP]
  9. 女神节,来聊聊这几位神一般的“程序媛”
  10. 融云 php web在线客户,GitHub - yy526063395/Web-IM-mini: PHP+layIM+融云简单实现版
  11. C# 基础-CLR-类型【0】
  12. geotools将shp数据存入postgres
  13. matlab输数据出结果,matlab对数据的输入输出
  14. Qt 周立功USBCAN总线上位机
  15. C#中MessageBox用法大全
  16. 排名前十名的WAP网站
  17. c语言long型是什么,c语言long类型是什么意思
  18. arcgis栅格缺值填补
  19. 笔记:Smith圆图及其计算
  20. 决策树用于股票分析整体介绍

热门文章

  1. Java链表数据结构刷题笔记总结
  2. java开发工具排名_排名前16的Java工具类
  3. Oracle coherence介绍
  4. MyBatis—引入外部配置文件(properties)
  5. 数据结构与算法(三):链表
  6. ps修改社保照片 不大于20KB
  7. 多个excel工作簿合并_EXCEL多表、多工作簿合并拆分,随心所欲(完善版)
  8. 关于excel表格直接引用和间接引用
  9. idea快速创建包快捷键大全_idea快捷键大全
  10. 新手自己搭建服务器步骤