深入研究Java类加载机制
类加载是Java程序运行的第一步,研究类的加载有助于了解JVM执行过程,并指导开发者采取更有效的措施配合程序执行。
研究类加载机制的第二个目的是让程序能动态的控制类加载,比如热部署等,提高程序的灵活性和适应性。
一、简单过程
Java程序运行的场所是内存,当在命令行下执行:
java HelloWorld
命令的时候,JVM会将HelloWorld.class加载到内存中,并形成一个Class的对象HelloWorld.class。
其中的过程就是类加载过程:
1、寻找jre目录,寻找jvm.dll,并初始化JVM;
2、产生一个Bootstrap Loader(引导类加载器);
3、Bootstrap Loader自动加载Extended Loader(标准扩展类加载器),并将其父Loader设为Bootstrap Loader。
4、Bootstrap Loader自动加载AppClass Loader(系统类加载器),并将其父Loader设为Extended Loader。
5、最后由AppClass Loader加载HelloWorld类。
以上就是类加载的最一般的过程。
图示:
二、各种类加载器解释:
1、Bootstrap Loader(引导类加载器):加载System.getProperty("sun.boot.class.path")所指定的路径或jar。
2、Extended Loader(标准扩展类加载器ExtClassLoader):加载System.getProperty("java.ext.dirs")所指定的路径或jar。在使用Java运行程序时,也可以指定其搜索路径,例如:java -Djava.ext.dirs=d:\projects\testproj\classes HelloWorld
3、AppClass Loader(系统类加载器AppClassLoader):加载System.getProperty("java.class.path")所指定的路径或jar。在使用Java运行程序时,也可以加上-cp来覆盖原有的Classpath设置,例如: java -cp ./lavasoft/classes HelloWorld
ExtClassLoader和AppClassLoader在JVM启动后,会在JVM中保存一份,并且在程序运行中无法改变其搜索路径。如果想在运行时从其他搜索路径加载类,就要产生新的类加载器。
三、类加载器的特点
1、运行一个程序时,总是由AppClass Loader(系统类加载器)开始加载指定的类。
2、在加载类时,每个类加载器会将加载任务上交给其父,如果其父找不到,再由自己去加载。(双亲委派机制)
3、Bootstrap Loader(引导类加载器)是最顶级的类加载器了,其父加载器为null.
3、Bootstrap Loader(引导类加载器)是最顶级的类加载器了,其父加载器为null.
四、类加载器的获取
很容易,看下面例子
public class HelloWorld { public static void main(String[] args) { HelloWorld hello = new HelloWorld(); Class c = hello.getClass(); ClassLoader loader = c.getClassLoader(); System.out.println(loader); System.out.println(loader.getParent()); System.out.println(loader.getParent().getParent()); } }打印结果:
sun.misc.Launcher$AppClassLoader@19821f sun.misc.Launcher$ExtClassLoader@addbf1 null Process finished with exit code 0从上面的结果可以看出,并没有获取到ExtClassLoader的父Loader,原因是Bootstrap Loader(引导类加载器)是用C语言实现的,找不到一个确定的返回父Loader的方式,于是就返回null。
五、类的加载
类加载有三种方式:
1、命令行启动应用时候由JVM初始化加载
2、通过Class.forName()方法动态加载
3、通过ClassLoader.loadClass()方法动态加载
三种方式区别比较大,看个例子就明白了:
public class HelloWorld { public static void main(String[] args) throws ClassNotFoundException { ClassLoader loader = HelloWorld.class.getClassLoader(); System.out.println(loader); //使用ClassLoader.loadClass()来加载类,不会执行初始化块 loader.loadClass("Test2"); //使用Class.forName()来加载类,默认会执行初始化块 // Class.forName("Test2"); //使用Class.forName()来加载类,并指定ClassLoader,初始化时不执行静态块 // Class.forName("Test2", false, loader); } }
public class Test2 { static { System.out.println("静态初始化块执行了!"); } }
分别切换加载方式,会有不同的输出结果。
六、Class.forName()和ClassLoader.loadClass()区别
Class.forName():将类的.class文件加载到jvm中之外,还会对类进行解释,执行类中的static块;
ClassLoader.loadClass():只干一件事情,就是将.class文件加载到jvm中,不会执行static中的内容,只有在newInstance才会去执行static块。
注:
Class.forName(name, initialize, loader)带参函数也可控制是否加载static块。并且只有调用了newInstance()方法采用调用构造函数,创建类的对象 。
示例:
1.使用classLoader加载
System.out.println("before loadClass... "); Class c =Test.class.getClassLoader().loadClass("com.hundsun.test.ClassInfo"); System.out.println("after loadClass... "); System.out.println("before newInstance... "); ClassInfo info1 =(ClassInfo) c.newInstance(); System.out.println("after newInstance... "); 输出结果: before loadClass... after loadClass... before newInstance... static invoked... contruct invoked... after newInstance...2.使用class.forName进行加载
System.out.println("before class.forName"); Class cc =Class.forName("com.hundsun.test.ClassInfo"); System.out.println("after class.forName"); ClassInfo info2 =(ClassInfo) cc.newInstance(); 输出结果: before class.forName static invoked... after class.forName before newInstance... contruct invoked... after newInstance...
下面说一下两者具体的执行过程 :
LoadClass()方法加载类及初始化过程:
类加载(loadclass())(加载)——》newInstance()(链接+初始化)
newInstance():
(开始连接)静态代码块——》普通变量分配准备(a=0;b=0;c=null)——》(开始初始化)普通变量赋值(a=1;b=2;c=”haha”)——》构造方法——》初始化成功。
Class.forName(Stirng className)一个参数方法加载类及初始化过程:
类加载(Class.forName())(加载)——》静态代码块——》newInstance()(链接+初始化)
newInstance():
(开始连接)普通变量分配准备(a=0;b=0;c=null)——》(开始初始化)普通变量赋值(a=1;b=2;c=”haha”)——》构造方法——》初始化成功。
Class.forName()三个参数的加载类及初始化过程同classLoader一样。
类加载(loadclass())(加载)——》newInstance()(链接+初始化)
newInstance():
(开始连接)静态代码块——》普通变量分配准备(a=0;b=0;c=null)——》(开始初始化)普通变量赋值(a=1;b=2;c=”haha”)——》构造方法——》初始化成功。
Class.forName(Stirng className)一个参数方法加载类及初始化过程:
类加载(Class.forName())(加载)——》静态代码块——》newInstance()(链接+初始化)
newInstance():
(开始连接)普通变量分配准备(a=0;b=0;c=null)——》(开始初始化)普通变量赋值(a=1;b=2;c=”haha”)——》构造方法——》初始化成功。
Class.forName()三个参数的加载类及初始化过程同classLoader一样。
从上边的断点调试可以看出,静态代码块不是在初始化阶段完成的,它陷于类初始化,先于普通变量默认分配(整型分配为0,字符串分配为null),这也就是为什么我们不能在静态代码块中引用普通变量的原因之一,这与上面所谓的“分配”、“初始化”相违背。
所以我觉得JVM的类加载及初始化过程应该是这样的:
1. 类加载:Bootstrap Loader——》Extended Loader——》System Loader 2. 静态代码块初始化 3. 链接: a) 验证:是否符合java规范 b) 准备:默认初始值 c) 解析:符号引用转为直接引用,解析地址 4. 初始化 a) 赋值:java代码中的初始值 b) 构造:构造函数
有关ClassLoader还有很重要一点:
同一个ClassLoader加载的类文件,只有一个Class实例。但是,如果同一个类文件被不同的ClassLoader载入,则会有两份不同的ClassLoader实例(前提是着两个类加载器不能用相同的父类加载器)。
ClassLoader源码分析:(双亲委派机制原理)
public Class<?> loadClass(String name)throws ClassNotFoundException { return loadClass(name, false); } protectedsynchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException { // 首先判断该类型是否已经被加载 Class c = findLoadedClass(name); if (c == null) { //如果没有被加载,就委托给父类加载或者委派给启动类加载器加载 try { if (parent != null) { //如果存在父类加载器,就委派给父类加载器加载 c = parent.loadClass(name, false); } else { //如果不存在父类加载器,就检查是否是由启动类加载器加载的类,通过调用本地方法native Class findBootstrapClass(String name) c = findBootstrapClass0(name); } } catch (ClassNotFoundException e) { // 如果父类加载器和启动类加载器都不能完成加载任务,才调用自身的加载功能 c = findClass(name); } } if (resolve) { resolveClass(c); } return c; }
相关推荐
### 深入研究Java类加载机制 #### 一、Java类加载机制概述 Java类加载机制是Java程序运行的第一步,它对于理解Java虚拟机(JVM)的行为至关重要。类加载过程涉及到类的加载、链接(验证、准备、解析)、初始化等...
深入研究Java的类加载机制 Java类加载机制是Java虚拟机的一项核心技术,可以在运行时刻动态地加载或替换系统的某些功能模块,而不影响系统其他功能模块的正常运行。类加载机制是Java虚拟机中的一项重要技术,可以使...
Java类加载机制是Java平台设计的核心部分之一,它负责将.class文件从磁盘或网络中读取并转换为运行时的Java对象。本篇将深入探讨Java类加载机制的细节,帮助开发者理解类如何被加载、连接以及初始化,以便更好地进行...
Java 类的动态装载机制是Java 虚拟机的一项核心技术,可以在运行时刻动态地加载或替换系统的 某些功能模块,而不影响系统其它功能模块的正常运行。介绍了Java 虚拟机中类的动态装载机制的原理、实现 及应用,分析了...
Tomcat,作为广泛使用的Java Servlet容器,它自定义了一套类加载机制,以满足Web应用程序的特殊需求。下面我们将深入探讨Java类加载器以及Tomcat中的类加载器。 在Java中,类加载器主要分为三个层次:Bootstrap ...
在Java Web开发中,Tomcat作为最常用的Servlet容器,其类加载机制对于理解和优化应用性能至关重要。本文将深入探讨Tomcat的ClassLoader是如何工作的,以及它如何影响到我们的应用程序。 首先,理解类加载器...
【Java类加载原理解析】 Java 类加载机制是Java技术体系中的重要组成部分,它负责将类的字节码文件从磁盘、网络或其他数据源加载到...通过深入研究类加载机制,我们能够提升对Java虚拟机和Java程序运行时行为的理解。
Java的类加载机制遵循双亲委派模型,即当一个类加载器接收到加载类的请求时,它会首先将这个任务委托给父类加载器,只有当父类加载器无法找到对应的类时,才会尝试自己去加载。这种设计有助于保持类的唯一性和避免...
Java类加载机制是Java运行时环境中的核心组成部分,它负责将类的字节码加载到JVM中并进行实例化。这个过程对于理解和优化Java应用性能至关重要。在本篇文章中,我们将深入探讨Java类加载器的工作原理,以及其在代码...
在Java编程语言中,类加载机制(ClassLoader)是理解JVM(Java Virtual Machine)工作原理的关键部分。它负责将.class文件从磁盘加载到内存中,使得Java程序能够执行。这个过程包括加载、验证、准备、解析和初始化五...
Java的反射机制是Java语言提供...要深入理解Java的反射机制,开发者需要对Java类的加载机制、JVM内存管理以及JDK提供的反射API有较为全面的了解。通过实际编程练习和对JDK源码的研究,可以更好地掌握反射的使用和原理。
总之,Java虚拟机是Java平台的核心,它的深入研究对于Java开发者来说极其重要,可以帮助他们更好地理解代码的运行机制,提升程序性能,以及解决运行时可能出现的问题。通过阅读如《Java虚拟机规范》这样的专业书籍,...
2. **类加载器机制**:类加载器是Java安全机制的重要组成部分。Java拥有层次化的类加载器体系,包括启动类加载器、标准扩展类加载器、路径加载器和网络类加载器。通过“双亲委派链模式”,类加载器能够确保只有可信...
总的来说,"classloader-playground"是一个实践和研究Java类加载机制的实用工具。通过这个项目,开发者不仅可以深入了解Java虚拟机的工作原理,还能掌握如何定制类加载器以满足特定场景的需求,提升系统的灵活性和可...
Dojo 是一个强大的JavaScript工具库,它为Web开发提供了丰富的功能和模块化系统。...通过对提供的源代码和文档的深入研究,我们可以更好地掌握Dojo的工作原理,从而在实际项目中充分利用这一特性。
这本书是Java深入研究的重要参考资料,涵盖了从基础到高级的JVM主题,对于提升Java编程技能和优化代码性能具有极大的价值。 首先,本书将详细介绍JVM的架构,包括类加载机制、内存模型以及执行引擎等核心部分。JVM...
研究Java虚拟机的解释执行机制,不仅有助于深入理解Java程序的执行过程,也为计算机研究人员和从业者提供了底层技术研究的契机。这有助于他们更好地掌握计算机技术,以及更高效地利用计算机工具进行软件开发。
在Tomcat中,类加载机制的设计是为了支持多个Web应用程序的隔离运行。每个Web应用都有自己的类加载器,这样可以确保一个应用的类不会意外地与另一个应用的类混淆。默认的类加载器结构包括以下层次: 1. Bootstrap ...
在Java开发过程中,为了提高开发效率和调试便捷性,我们经常需要实现类的热替换...这个压缩包中的源码可能就是展示如何实现这个功能的一个实例,你可以仔细研究这些代码,加深对Java类加载机制和热替换技术的理解。