热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

深入JVMClass装载系统

一、Class文件的装载过程Class类型通常以文件的形式存在(当然,任何二进制流都可以是Class类型),只有被Java虚拟机装载的Class类型才能

一、Class文件的装载过程

Class类型通常以文件的形式存在(当然,任何二进制流都可以是Class类型),只有被Java虚拟机装载的Class类型才能在程序中使用。系统状态Class类型可以分为加载、连接和初始化3个步骤。其中,连接又可分为验证、准备和解析3步。

1.1 类装载的条件

Class只有在必须要使用的时候才会被装载,Java虚拟机不会无条件的装载Class类型。Java虚拟机规定,一个类或接口在初次使用前,必须要进行初始化。这里指的“使用”,是指主动使用,主动使用只有下列几种情况:

  • 创建一个类的实例,比如使用new关键字,或者通过反射、克隆、反序列化。
  • 使用类的静态方法时,即当使用了字节码invokestatic指令。
  • 使用类或接口的静态字段(final常量除外),比如,使用getstatic或者putstatic指令。
  • 使用java.lang.reflect包中的方法反射类的方法时。
  • 当初始化子类时,要求先初始化父类。
  • 作为启动虚拟机,含义main()方法的那个类。

除了以上的情况属于主动使用,其他的情况均属于被动使用。被动使用不会引起类的初始化。

主动引用的例子:

public class Parent {static{System.out.println("Parent init");}
}public class Child extends Parent{static {System.out.println("Child init");}
}public class IninMain{public static void main(String[] args){Child c = new Child();}
}

执行InitMain,结果为:

Parent init
Child init

被动引用的例子,被动引用不会导致类的装载。

public class Parent{static{System.out.println("Parent init");}public static int v = 100;
}public class Child extends Parent{static{System.out.println("Child init");}
}public class UseParent{public static void main(String[] args){System.out.println(Child.v);}
}

输出结果为:

Parent init
100

可以看到,虽然在UseParent中,直接访问了子类对象,但是Child子类并未被初始化,只有Parent父类被初始化。ke'jian可见,在引用一个字段时,只有直接定义该字段的类,才会被初始化。

final常量不会引起类的初始化。

public class FinalFieldClass {public static final String constString = "CONST";static {System.out.println("FinalFieldClass init");}
}public class UseFinalField {public static void main(String[] args){System.out.println(FinalFieldClass.constString);}
}

运行以上代码输出结果为:

CONST

1.2 加载类

加载类处于类加载的第一个阶段。在加载类时,Java虚拟机必须完成以下工作:

  • 通过类的全名,获取类的二进制数据流。
  • 解析类的二进制数据流为方法区内的数据结构。
  • 创建java.lang.Class类的实例,表示该类型。
  • 1.3 验证类

Java虚拟机验证过程

1.4 准备

当一个类验证通过时,虚拟机就会进入准备阶段。在这个阶段,虚拟机会为这个类分配相应的内存空间,并设置初始值。

1.5 解析类

在准备阶段完成后,就进入了解析阶段。解析阶段的工作就是将类、接口、字段和方法的符号引用转为直接引用。

1.6 初始化

类的初始化是类装载的最后一个阶段。如果前面的步骤都没有问题,那么表示类可以顺利装载到系统中。此时,类才会开始执行Java字节码。初始化阶段的重要工作时执行类的初始化方法。方法是由编译器自动生成的,它是由类静态成员的赋值语句以及static语句块合并产生的。

二、掌握ClassLoader

2.1 看懂类加载

ClassLoader是Java的核心组件,所有的Class都是由ClassLoader进行加载的,ClassLoader负责通过各种方式将Class信息的二进制数据流读入系统,然后交给Java虚拟机进行连接、初始化等操作。因此,ClassLoader在整个装在阶段,只能影响到类的加载,而无法通过ClassLoader去改变类的连接和初始化行为。

从代码层面看,ClassLoader是一个抽象类,它提供了一些重要的接口,用于自定义Class的加载流程和加载方式。

在ClassLoader的结构中,还有一个重要的字段parent,他也是一个ClassLoader的实例,这个字段锁表示的ClassLoader也称为这个ClassLoader的双亲。在类加载的过程中,ClassLoader可能会将某些请求交与自己的双亲处理。

2.2 ClassLoader的分类

在标准的Java程序中,Java虚拟机会创建3类ClassLoader为整个应用程序服务。他们分别是:Bootstrap ClassLoader(启动类加载器)、Extension ClassLoader(扩展类加载器)和App ClassLoader(应用类加载器,也称为系统类加载器)。此外,每个应用程序还可以拥有自定义的ClassLoader,扩展Java虚拟机获取Class数据的能力。

ClassLoader层次结构

在虚拟机设计中,使用这种分散的ClassLoader去装载类是有好处的,不同层次的类可以由不同的ClassLoader加载,从而进行划分,这有助于系统的模块化设计。一般来说,启动类加载器负责加载系统的核心类,比如rt.jar中的Java类;扩展类加载器用于加载%JAVA_HOME%/lib/ext/*.jar中的Java类;应用类加载器用于加载用户类,也就是用户程序的类;自定义类加载器用于加载一些特殊途径的类,一般也是用户程序类。


转载于:https://www.cnblogs.com/f-zhao/p/6196563.html


推荐阅读
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 原文地址:https:www.cnblogs.combaoyipSpringBoot_YML.html1.在springboot中,有两种配置文件,一种 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • Java中包装类的设计原因以及操作方法
    本文主要介绍了Java中设计包装类的原因以及操作方法。在Java中,除了对象类型,还有八大基本类型,为了将基本类型转换成对象,Java引入了包装类。文章通过介绍包装类的定义和实现,解答了为什么需要包装类的问题,并提供了简单易用的操作方法。通过本文的学习,读者可以更好地理解和应用Java中的包装类。 ... [详细]
  • 欢乐的票圈重构之旅——RecyclerView的头尾布局增加
    项目重构的Git地址:https:github.comrazerdpFriendCircletreemain-dev项目同步更新的文集:http:www.jianshu.comno ... [详细]
  • Android系统源码分析Zygote和SystemServer启动过程详解
    本文详细解析了Android系统源码中Zygote和SystemServer的启动过程。首先介绍了系统framework层启动的内容,帮助理解四大组件的启动和管理过程。接着介绍了AMS、PMS等系统服务的作用和调用方式。然后详细分析了Zygote的启动过程,解释了Zygote在Android启动过程中的决定作用。最后通过时序图展示了整个过程。 ... [详细]
author-avatar
gl6474177
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有