`

Java虚拟机类加载机制浅谈

 
阅读更多

http://computerdragon.blog.51cto.com/6235984/1223354

 

 

   Java语言是一种编译后再经过解释器执行的过程, 解释器主要就是如何处理解释Class文件的二进制字节流。JVM主要包含三大核心部分:运行时数据区,类加载器和执行引擎。

       虚拟机将描述类的数据从Class文件加载到内存,并对数据进行校验、准备、解析和初始化,最终就会形成可以被虚拟机使用的Java类型,这就是一个虚拟机的类加载机制。Java中的类是动态加载的,只有在运行期间使用到该类的时候,才会将该类加载到内存中,Java依赖于运行期动态加载和动态链接来实现类的动态使用。

一个类的整个生命周期如下:

    加载,验证,准备,初始化和卸载在开始的顺序上是固定的,但是可以交叉进行。

   在Java中,对于类有且仅有四种情况会对类进行“初始化”。

     1) 使用new关键字实例化对象的时候,读取或设置一个类的静态字段时候(除final修饰的static外),调用类的静态方法时候,都只会初始化该静态字段或者静态方法所定义的类。

     2) 使用reflect包对类进行放射调用的时候,如果类没有进行初始化,则先要初始化该类

     3) 当初始化一个类的时候,如果其父类没有初始化过,则先要触发其父类初始化。

     4)  虚拟机启动的时候,会初始化一个有main方法的主类。

注意:通过子类引用父类静态字段,只会初始化父类不会初始化子类;通过数组定义来引用类,也不会触发该类的初始化;常量在编译阶段会存入调用类的常量池中,本质上没有直接引用到定义常量的类,因此也不会触发定义常量的类的初始化。

一 类加载过程

1 加载

       加载阶段主要完成三件事,即通过一个类的全限定名来获取定义此类的二进制字节流,将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构,在Java堆中生成一个代表此类的Class对象,作为访问方法区这些数据的入口。这个加载过程主要就是靠类加载器实现的,这个过程可以由用户自定义类的加载过程。

2 验证

这个阶段目的在于确保Class文件的字节流中包含信息符合当前虚拟机要求,不会危害虚拟机自身安全。主要包括四种验证:

     文件格式验证:基于字节流验证,验证字节流是否符合Class文件格式的规范,并且能被当前虚拟机处理。

     元数据验证:基于方法区的存储结构验证,对字节码描述信息进行语义验证。

     字节码验证:基于方法区的存储结构验证,进行数据流和控制流的验证。

     符号引用验证:基于方法区的存储结构验证,发生在解析中,是否可以将符号引用成功解析为直接引用。

3 准备

仅仅为类变量(即static修饰的字段变量)分配内存并且设置该类变量的初始值即零值,这里不包含用final修饰的static,因为final在编译的时候就会分配了,同时这里也不会为实例变量分配初始化。类变量会分配在方法区中,而实例变量是会随着对象一起分配到Java堆中。

4 解析

解析主要就是将常量池中的符号引用替换为直接引用的过程。符号引用就是一组符号来描述目标,可以是任何字面量,而直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。有类或接口的解析,字段解析,类方法解析,接口方法解析。

 这里要注意如果有一个同名字段同时出现在一个类的接口和父类中,那么编译器一般都会拒绝编译。

5 初始化

    初始化阶段依旧是初始化类变量和其他资源,这里将执行用户的static字段和静态语句块的赋值操作。这个过程就是执行类构造器<clinit>方法的过程。

<clinit>方法是由编译器收集类中所有类变量的赋值动作和静态语句块的语句生成的,类构造器<clinit>方法与实例构造器<init>方法不同,这里面不用显示的调用父类的<clinit>方法,父类的<clinit>方法会自动先执行于子类的<clinit>方法。即父类定义的静态语句块和静态字段都要优先子类的变量赋值操作。

二 类加载器

1   类加载器的分类

       启动类加载器(Bootstrap ClassLoader):  主要负责加载<JAVA_HOME>\lib目录中的,或是-Xbootclasspath参数指定的路径中的,并且可以被虚拟机识别(仅仅按照文件名识别的)的类库到虚拟机内存中。它加载的是System.getProperty("sun.boot.class.path")所指定的路径或jar。

       扩展类加载器(Extension ClassLoader):主要负责加载<JAVA_HOME>\lib\ext目录中的,或者被java.ext.dirs系统变量所指定的路径中的所有类库。它加载的是

System.getProperty("java.ext.dirs")所指定的路径或jar。

       应用程序类加载器(Application ClassLoader):也叫系统类加载器,主要负责加载ClassPath路径上的类库,如果应用程序没有自定义自己类加载器,则这个就是默认的类加载器。

它加载的是System.getProperty("java.class.path")所指定的路径或jar。

 

2   类加载器的特点

1)运行一个程序时,总是由Application Loader(系统类加载器)开始加载指定的类。

2)在加载类时,每个类加载器会将加载任务上交给其父,如果其父找不到,再由自己去加载。

3)Bootstrap Loader(启动类加载器)是最顶级的类加载器了,其父加载器为null.

3   类加载器的双亲委派模型

       类加载器双亲委派模型的工作过程是:如果一个类加载器收到一个类加载的请求,它首先将这个请求委派给父类加载器去完成,每一个层次类加载器都是如此,则所有的类加载请求都会传送到顶层的启动类加载器,只有父加载器无法完成这个加载请求(即它的搜索范围中没有找到所要的类),子类才尝试加载。

      使用双亲委派模型主要是两个原因:1)可以避免重复加载,当父类已经加载了,则就子类不需再次加载;2)安全因素,如果不用这种,则用户可以随意的自定义加载器来替代Java核心API,则就会带来安全隐患。

       下面是一个类加载器双亲委派模型,这里各个类加载器并不是继承关系,它们利用组合实现的父类与子类关系。

 

4 类加载的几种方式

 

1) 命令行启动应用时候由JVM初始化加载,加载含有main的主类。

      2)通过Class.forName("Hello")方法动态加载类,默认会执行初始化块,这是因为Class.forName("Hello")其实就是Class.forName("Hello",true,CALLCLASS.getClassLoader()),第二个参数就是类加载过程中的连接操作。如果指定了ClassLoader,则不会执行初始化块。

      3)通过ClassLoader.loadClass("Hello")方法动态加载类,不会执行初始化块,因为loadClass方法有两个参数,用户只是用第一个参数,第二个参数默认为false,即不对该类进行解析则就不会初始化。

5 类加载实例

 

当在命令行下执行:java HelloWorld(HelloWorld是含有main方法的类的Class文件),JVM会将HelloWorld.class加载到内存中,并在堆中形成一个Class的对象HelloWorld.class。

      基本的加载流程如下:

      1)寻找jre目录,寻找jvm.dll,并初始化JVM;

      2)产生一个Bootstrap Loader(启动类加载器);

      3)Bootstrap Loader,该加载器会加载它指定路径下的Java核心API,并且再自动加载Extended Loader(标准扩展类加载器),Extended Loader会加载指定路径下的扩展JavaAPI,并将其父Loader设为BootstrapLoader。

      4)Bootstrap Loader也会同时自动加载AppClass Loader(系统类加载器),并将其父Loader设为ExtendedLoader。

     5)最后由AppClass Loader加载CLASSPATH目录下定义的类,HelloWorld类。

分享到:
评论

相关推荐

    浅谈JAVA虚拟机JVM及工作原理

    "浅谈JAVA虚拟机JVM及工作原理" Java虚拟机(JVM)是Java语言的 runtime 环境,它提供了一个平台独立的环境,使得Java程序可以跨平台运行。JVM 的主要组件包括虚拟机栈、堆、方法区、程序计数器、本地方法栈等。 1...

    java培训教程-浅谈java虚拟机.pdf

    Java虚拟机(JVM)是Java编程语言的核心组成部分,它为Java程序提供了跨平台的运行环境。Java的“一次编写,到处运行”特性主要得益于JVM的存在。在JVM内部,程序被编译成字节码,这是一种平台无关的中间表示,可以...

    Java反射技术浅谈.pdf

    JVM(Java虚拟机)能够将类加载到内存中,并提供一套反射机制来访问类的信息。Java提供了多种方法来访问类的信息,包括但不限于`getMethod()`、`getField()`、`getConstructor()`等。这些方法允许程序员在运行时获取...

    浅谈Java的虚拟机结构以及虚拟机内存的优化

    Java虚拟机(JVM)是Java程序的核心组成部分,它负责运行字节码,管理内存,以及执行垃圾回收等关键任务。本文将简要介绍JVM的结构,特别是内存模型,以及如何进行内存优化。 首先,JVM由多个子系统组成,其中类装载...

    浅谈两个jar包中包含完全相同的包名和类名的加载问题

    Java中的类加载机制是指Java虚拟机(JVM)在加载类文件时的相关处理过程。当多个Jar包中包含相同的包名和类名时,可能会出现类加载问题。下面我们来讨论这种情况下的类加载问题。 类加载机制 Java中的类加载机制是...

    浅谈计算机软件开发的JAVA编程语言.zip

    Java的跨平台能力得益于Java虚拟机(JVM)。JVM允许Java程序在任何安装了相应JVM的平台上运行,这使得开发者可以编写一次代码,无需修改就能在不同的操作系统上部署,极大地提高了开发效率。JVM还负责解释和优化Java...

    浅谈jvm原理

    "浅谈 JVM 原理" JVM(Java Virtual Machine)是一种虚拟机,它可以模拟完整的硬件系统功能,运行在一个完全隔离的环境中,提供了一个完整的计算机系统。JVM 可以分为三类:VMWare、Visual Box 和 JVM。其中,...

    浅谈为什么java命令运行class文件出现异常共7页.p

    当我们在命令行输入`java`命令时,实际上是在调用JVM(Java虚拟机)来执行字节码文件(.class)。这个过程包括以下步骤: 1. **编译阶段**:Java源代码(.java文件)通过javac命令被编译成字节码文件(.class文件)...

    浅谈计算机软件开发的JAVA编程语言.pdf

    其次,JAVA语言在编程模式上偏向于对象导向,与C++等语言相比,JAVA更注重于对象的使用,并且在编译时生成二进制字节码,这在执行时通过Java虚拟机进行解释,这种编译机制使得JAVA语言的应用更为广泛和安全。...

    浅谈JVM核心之JVM运行和类加载

    在Java世界中,虚拟机(JVM)...总的来说,了解JVM的运行时数据区和类加载机制对于优化Java应用程序性能、解决类冲突和理解类加载过程至关重要。通过深入学习这些概念,开发者可以更好地理解和控制Java程序的运行行为。

    浅谈Java内存区域划分和内存分配策略

    "浅谈Java内存区域划分和内存分配策略" 本文将详细讲述Java内存区域划分和内存分配策略,涵盖程序计数器、虚拟机栈、本地方法栈、堆、方法区等内存区域的概念和作用,以及对象创建过程和内存分配策略。 程序计数器...

    java基础PPT

    11. **JNI与JVM原理**:浅谈Java Native Interface(JNI),用于在Java程序中调用本地(非Java)代码,以及JVM的工作原理,包括类加载、内存管理和垃圾收集。 12. **案例分析**:可能包含一些简单的编程实例,帮助...

    浅谈Android Classloader动态加载分析

    Android中的ClassLoader机制是Java应用程序运行的核心组成部分,它负责查找并加载类文件到Java虚拟机(JVM)中,使得程序能够执行相应的代码。在Android系统中,ClassLoader的使用与Java环境有所不同,主要体现在...

    浅谈计算机应用软件开发中编程语言的选择研究中英文对照.pdf

    浅谈计算机应用软件开发中编程语言的选择研究 随着信息技术的飞速发展,计算机应用软件已经深入到我们生活的各个领域,从个人娱乐、办公自动化到工业自动化、云计算等,无处不在。软件开发企业面临着激烈的市场竞争...

    java高手的文章合集1/3

    10. **Java虚拟机(JVM)**:讲解JVM的工作原理,包括类加载机制、内存模型(堆、栈、方法区等)、垃圾回收以及性能优化策略。 11. **注解(Annotation)**:介绍注解的定义、元注解和自定义注解的使用,以及在编译...

Global site tag (gtag.js) - Google Analytics