登录
注册
开源
企业版
高校版
搜索
帮助中心
使用条款
关于我们
开源
企业版
高校版
私有云
模力方舟
AI 队友
登录
注册
代码拉取完成,页面将自动刷新
捐赠
捐赠前请先登录
取消
前往登录
扫描微信二维码支付
取消
支付完成
支付提示
将跳转至支付宝完成支付
确定
取消
Watch
不关注
关注所有动态
仅关注版本发行动态
关注但不提醒动态
24
Star
56
Fork
2
Java技术交流
/
Java技术提升库
代码
Issues
56
Pull Requests
0
Wiki
统计
流水线
服务
JavaDoc
PHPDoc
质量分析
Jenkins for Gitee
腾讯云托管
腾讯云 Serverless
悬镜安全
阿里云 SAE
Codeblitz
SBOM
我知道了,不再自动展开
更新失败,请稍后重试!
移除标识
内容风险标识
本任务被
标识为内容中包含有代码安全 Bug 、隐私泄露等敏感信息,仓库外成员不可访问
issue_3167283
待办的
#I1VVW3
Java老郑
成员
创建于
2020-09-19 12:06
内容可能含有违规信息
### 考点分析! 今天的问题是关于JVM类加载方面的基础问题,我前面给出的回答参考了Java 虚拟机规范中的主要条款。如果你在面试中回答这个问题,在这个基础上还可以举例说明。 我们来看一个经典的延伸问题,准备阶段谈到静态变量,那么对于常量和不同静态变量有什么区别? 需要明确的是,没有人能够精确的理解和记忆所有信息,如果碰到这种问题,有直接答案当然最好;没有的话,就说说自己的思路。 我们定义下面这样的类型,分别提供了普通静态变量、静态常量,常量又考虑到原始类型和引用类型可能有区别。 ``` public class CLPreparation { public static int a= 100; public static final int INT_CoNSTANT = 190e; pubic static final Integer INTEER_CONSTANT= Integer.valueof(1900); } ``` 编译并反编译一下∶ ``` Javac CLPreparation.java Javap -v CLPreparation.class ``` 可以在字节码中看到这样的额外初始化逻辑∶ ``` 0:bipush 100 2:putstatic #2 5:sipush 1000 8:invokestatic #3 11: putstatic #4 ``` 这能让我们更清楚,普通原始类型静态变量和引用类型(即使是常量),是需要额外调用putstatic等VM指令的,这些是在显式初始化阶段执行,而不是准备阶段调用;而原始类型常量,则不需要这样的步骤。 关于类加载过程的更多细节,有非常多的优秀资料进行介绍,你可以参考大名鼎鼎的《深入理解Java 虚拟机》,一本非常好的入门书籍。我的建议是不要仅看教程,最好能够想出代码实例去验证自己对某个方面的理解和判断,这样不仅能加深理解,还能够在未来的应用开发中使用到。 其实,类加载机制的范围实在太大,我从开发和部署的不同角度,各选取了一个典型扩展问题供你参考∶ - 如果要真正理解双亲委派模型,需要理解 Java 中类加载器的架构和职责,至少要懂具体有哪些内建的类加载器,这些是我上面的回答里没有提到的;以及如何自定义类加载器? - 从应用角度,解决某些类加载问题,例如我的Java程序启动较慢,有没有办法尽量减小Java类加载的开销 ? 另外,需要注意的是,在Java9中,Jigsaw项目为 Java提供了原生的模块化支持,内建的类加载器结构和机制发生了明显变化。我会对此进行讲解,希望能够避免一些未来升级中可能发生的问题。 ### 问题回答! 一般来说,我们把Java的类加载过程分为三个主要步骤∶加载、链接、初始化,具体行为在ava虚拟机规范里有非常详细的定义。 首先是加载阶段(Loading),它是Java将字节码数据从不同的数据源读取到JVM中,并映射为JⅣM认可的数据结构(Class对象),这里的数据源可能是各种各样的形态,如jr文件、dlass文件,甚至是网络数据源等;如果输入数据不是ClassFile的结构,则会抛出ClassFormatError。 加载阶段是用户参与的阶段,我们可以自定义类加载器,去实现自己的类加载过程。 第二阶段是链接(Linking),这是核心的步骤,简单说是把原始的类定义信息平滑地转化入ⅣM 运行的过程中。这里可进一步细分为三个步骤∶ - 验证(Verification),这是虚拟机安全的重要保障,JVM需要核验字节信息是符合Java虚拟机规范的,否则就被认为是VerifyError,这样就防止了恶意信息或者不合规的信息危害JVM的运行,验证阶段有可能触发更多 class的加载。 - 准备(Preparation),创建类或接口中的静态变量,并初始化静态变量的初始值。但这里的"初始化"和下面的显式初始化阶段是有区别的,侧重点在于分配所需要的内存空间,不会去执行更进一步的JVM指令。 - 解析(Resolution),在这一步会将常量池中的符号引用(symbolicreference)替换为直接引用。在ava虚拟机规范中,详细介绍了类、接口、方法和字段等各个方面的解析。 最后是初始化阶段(initalization),这一步真正去执行类初始化的代码逻辑,包括静态字段赋值的动作,以及执行类定义中的静态初始化块内的逻辑,编译器在编译阶段就会把这部分逻辑整理好,父类型的初始化逻辑优先于当前类型的逻辑。 再来谈谈双亲委派模型,简单说就是当类加载器(Class-Loader)试图加载某个类型的时候,除非父加载器找不到相应类型,否则尽量将这个任务代理给当前加载器的父加载器去做。使用委派模型的目的是避免重复加载 Java类型。 ### 后续扩展! 首先,从架构角度,一起来看看 Java 8以前各种类加载器的结构,下面是三种Oracle JDK 内建的类加载器。 - 启动类加载器(Bootstrap Class-Loader),加载jre/lib 下面的jar文件,如 rtjar。它是个超级公民,即使是在开启了 Security Manager的时候,JDK 仍赋予了它加载的程序 AIPermission。 对于做底层开发的工程师,有的时候可能不得不去试图修改JDK的基础代码,也就是通常意义上的核心类库,我们可以使用下面的命令行参数。 ``` # 指定新的 bootclasspath,替换 java.*包的内部实现 java-Xbootclasspath:<your_boot_classpath> your_App # a 意味着 append,将指定目录添加到bootclasspath 后面 java -Xbootclasspath/a:<your_dir> your_App # p 意味着 prepend,将指定目录添加到 bootclasspath前面 java -Xbootclasspath/p:<your_dir> your_App ``` 用法其实很易懂,例如,使用最常见的"/p",既然是前置,就有机会替换个别基础类的实现。 我们一般可以使用下面方法获取父加载器,但是在通常的JDK/JRE 实现中,扩展类加载器getParent()都只能返回 null。 ``` public final ClassLoader getParent() ``` - 扩展类加载器(Extension or Ext Class-Loader),负责加载我们放到jre/lib/ext/目录下面的jar包,这就是所谓的extension机制。该目录也可以通过设置"java.ext.dirs"来覆盖。 ``` java -Djava.ext.dirs=your_ext_dir Helloworld ``` - 应用类加载器(Application or App Class-Loader),就是加载我们最熟悉的 classpath的 内容。这里有一个容易混淆的概念,系统(System)类加载器,通常来说,其默认就是JDK 内建的应用类加载器,但是它同样是可能修改的,比如∶ ``` java-Djava.system.class.loader=com.yourcorp.YourClassLoader HelloWorld ``` 如果我们指定了这个参数,JDK 内建的应用类加载器就会成为定制加载器的父亲,这种方式通常用在类似需要改变双亲委派模式的场景。 具体请参考下图∶  至于前面被问到的双亲委派模型,参考这个结构图更容易理解。试想,如果不同类加载器都自己加载需要的某个类型,那么就会出现多次重复加载,完全是种浪费。 通常类加载机制有三个基本特征∶ - 双亲委派模型。但不是所有类加载都遵守这个模型,有的时候,启动类加载器所加载的类型,是可能要加载用户代码的,比如JDK 内部的 ServiceProvider/Seviceloade机制,用户可以在标准 API框架上,提供自己的实现,JDK也需要提供些默认的参考实现。例如,Java中 JNDI、JDBC、文件系统、Cipher 等很多方面,都是利用的这种机制,这种情况就不会用双亲委派模型去加载,而是利用所谓的上下文加载器。 - 可见性,子类加载器可以访问父加载器加载的类型,但是反过来是不允许的,不然,因为缺少必要的隔离,我们就没有办法利用类加载器去实现容器的逻辑。 - 单一性,由于父加载器的类型对于子加载器是可见的,所以父加载器中加载过的类型,就不会在子加载器中重复加载。但是注意,类加载器"邻居"间,同一类型仍然可以被加载多次,因为互相并不可见。 在 JDK9中,由于Jigsaw项目引入了Java平台模块化系统(JPMS),Java SE的源代码被划分为一系列模块。  类加载器,类文件容器等都发生了非常大的变化,我这里总结一下∶ - 前面提到的-Xbootclasspath参数不可用了。API已经被划分到具体的模块,所以上文中,利用"-Xbootclasspath/p"替换某个Java核心类型代码,实际上变成了对相应的模块进行的修补,可以采用下面的解决方案∶ 首先,确认要修改的类文件已经编译好,并按照对应模块(假设是java.base)结构存放,然后,给模块打补丁∶ ``` java --patch-module java.base=your_patch yourApp ``` - 扩展类加载器被重命名为平台类加载器(Platform Class-Loader),而且extension机制则被移除。也就意味着,如果我们指定java.ext.dirs 环境变量,或者 lib/ext目录存在,JVM 将直接返回错误!建议解决办法就是将其放入 classpath 里。 - 部分不需要AIPermission的Java 基础模块,被降级到平台类加载器中,相应的权限也被更精细粒度地限制起来。 - rtjar和 toolsjar同样是被移除了!JDK的核心类库以及相关资源,被存储在 jimage文件中,并通过新的 JRT文件系统访问,而不是原有的JAR 文件系统。虽然看起来很惊人,但幸好对于大部分软件的兼容性影响,其实是有限的,更直接地影响是 IDE 等软件,通常只要升级到新版本就可以了。 - 增加了Layer的抽象,VM启动默认创建 BootLayer,开发者也可以自己去定义和实例化Layer,可以更加方便的实现类似容器一般的逻辑抽象。 结合了Layer,目前的VM内部结构就变成了下面的层次,内建类加载器都在 BootLayer 中,其他Layer 内部有自定义的类加载器,不同版本模块可以同时工作在不同的Layer。  谈到类加载器,绕不过的一个话题是自定义类加载器,常见的场景有∶ - 实现类似进程内隔离,类加载器实际上用作不同的命名空间,以提供类似容器、模块化的效果。例如,两个模块依赖于某个类库的不同版本,如果分别被不同的容器加载,就可以互不干扰。这个方面的集大成者是ava E和OSGl、PMS等框架。 - 应用需要从不同的数据源获取类定义信息,例如网络数据源,而不是本地文件系统。 - 或者是需要自己操纵字节码,动态修改或者生成类型。我们可以总体上简单理解自定义类加载过程∶ - 通过指定名称,找到其二进制实现,这里往往就是自定义类加载器会"定制"的部分,例如,在特定数据源根据名字获取字节码,或者修改或生成字节码。 - 然后,创建 Class对象,并完成类加载过程。二进制信息到lass对象的转换,通常就依赖defineClass,我们无需自己实现,它是final 方法。有了Class对象,后续完成加载过程就顺理成章了。 今天我梳理了一下类加载的过程,并针对Java新版中类加载机制发生的变化,进行了相对全面的总结,最后介绍了一个改善类加载速度的特性,希望对你有所帮助。
评论 (
2
)
登录
后才可以发表评论
状态
待办的
待办的
进行中
已完成
已关闭
负责人
未设置
赵伟风
zhao-weifeng
负责人
协作者
+负责人
+协作者
标签
未设置
标签管理
里程碑
01.JavaSE阶段面试题
未关联里程碑
Pull Requests
未关联
未关联
关联的 Pull Requests 被合并后可能会关闭此 issue
分支
未关联
未关联
master
开始日期   -   截止日期
-
置顶选项
不置顶
置顶等级:高
置顶等级:中
置顶等级:低
优先级
不指定
严重
主要
次要
不重要
参与者(1)
1
https://gitee.com/beike-java-interview-alliance/java-interview.git
git@gitee.com:beike-java-interview-alliance/java-interview.git
beike-java-interview-alliance
java-interview
Java技术提升库
点此查找更多帮助
搜索帮助
Git 命令在线学习
如何在 Gitee 导入 GitHub 仓库
Git 仓库基础操作
企业版和社区版功能对比
SSH 公钥设置
如何处理代码冲突
仓库体积过大,如何减小?
如何找回被删除的仓库数据
Gitee 产品配额说明
GitHub仓库快速导入Gitee及同步更新
什么是 Release(发行版)
将 PHP 项目自动发布到 packagist.org
评论
仓库举报
回到顶部
登录提示
该操作需登录 Gitee 帐号,请先登录后再操作。
立即登录
没有帐号,去注册