《深入理解java虚拟机》学习笔记6——类加载机制

转载 2013年12月04日 15:34:44

Java虚拟机类加载过程是把Class类文件加载到内存,并对Class文件中的数据进行校验、转换解析和初始化,最终形成可以被虚拟机直接使用的java类型的过程。

在加载阶段,java虚拟机需要完成以下3件事:

a.通过一个类的全限定名来获取定义此类的二进制字节流。

b.将定义类的二进制字节流所代表的静态存储结构转换为方法区的运行时数据结构。

c.在java堆中生成一个代表该类的java.lang.Class对象,作为方法区数据的访问入口。

Java虚拟机的类加载是通过类加载器实现的, Java中的类加载器体系结构如下:

(1).BootStrap ClassLoader:启动类加载器,负责加载存放在%JAVA_HOME%\lib目录中的,或者通被-Xbootclasspath参数所指定的路径中的,并且被java虚拟机识别的(仅按照文件名识别,如rt.jar,名字不符合的类库,即使放在指定路径中也不会被加载)类库到虚拟机的内存中,启动类加载器无法被java程序直接引用。

(2).Extension ClassLoader:扩展类加载器,由sun.misc.Launcher$ExtClassLoader实现,负责加载%JAVA_HOME%\lib\ext目录中的,或者被java.ext.dirs系统变量所指定的路径中的所有类库,开发者可以直接使用扩展类加载器。

(3).Application ClassLoader:应用程序类加载器,由sun.misc.Launcher$AppClassLoader实现,负责加载用户类路径classpath上所指定的类库,是类加载器ClassLoader中的getSystemClassLoader()方法的返回值,开发者可以直接使用应用程序类加载器,如果程序中没有自定义过类加载器,该加载器就是程序中默认的类加载器。

注意:上述三个JDK提供的类加载器虽然是父子类加载器关系,但是没有使用继承,而是使用了组合关系。

从JDK1.2开始,java虚拟机规范推荐开发者使用双亲委派模式(ParentsDelegation Model)进行类加载,其加载过程如下:

(1).如果一个类加载器收到了类加载请求,它首先不会自己去尝试加载这个类,而是把类加载请求委派给父类加载器去完成。

(2).每一层的类加载器都把类加载请求委派给父类加载器,直到所有的类加载请求都应该传递给顶层的启动类加载器。

(3).如果顶层的启动类加载器无法完成加载请求,子类加载器尝试去加载,如果连最初发起类加载请求的类加载器也无法完成加载请求时,将会抛出ClassNotFoundException,而不再调用其子类加载器去进行类加载。

双亲委派 模式的类加载机制的优点是java类它的类加载器一起具备了一种带优先级的层次关系,越是基础的类,越是被上层的类加载器进行加载,保证了java程序的稳定运行。双亲委派模式的实现:
[java] view plaincopy
  1. protected synchronized Class<?> loadClass(String name, Boolean resolve) throws ClassNotFoundException{  
  2.     //首先检查请求的类是否已经被加载过  
  3.     Class c = findLoadedClass(name);  
  4.     if(c == null){  
  5.     try{  
  6.         if(parent != null){//委派父类加载器加载  
  7.     c = parent.loadClass(name, false);  
  8. }  
  9. else{//委派启动类加载器加载  
  10.     c = findBootstrapClassOrNull(name);   
  11. }  
  12. }catch(ClassNotFoundException e){  
  13.     //父类加载器无法完成类加载请求  
  14. }  
  15. if(c == null){//本身类加载器进行类加载  
  16.     c = findClass(name);  
  17. }  
  18. }  
  19. if(resolve){  
  20.     resolveClass(c);  
  21. }  
  22. return c;  
  23. }  

若要实现自定义类加载器,只需要继承java.lang.ClassLoader 类,并且重写其findClass()方法即可。java.lang.ClassLoader 类的基本职责就是根据一个指定的类的名称,找到或者生成其对应的字节代码,然后从这些字节代码中定义出一个 Java 类,即 java.lang.Class 类的一个实例。除此之外,ClassLoader 还负责加载 Java 应用所需的资源,如图像文件和配置文件等,ClassLoader 中与加载类相关的方法如下:

方法

说明

getParent()

返回该类加载器的父类加载器。

loadClass(String name)

加载名称为 二进制名称为name 的类,返回的结果是 java.lang.Class 类的实例。

 

findClass(String name)

查找名称为 name 的类,返回的结果是 java.lang.Class 类的实例。

 

findLoadedClass(String name)

查找名称为 name 的已经被加载过的类,返回的结果是 java.lang.Class 类的实例。

 

resolveClass(Class<?> c)

链接指定的 Java 类。

注意:在JDK1.2之前,类加载尚未引入双亲委派模式,因此实现自定义类加载器时常常重写loadClass方法,提供双亲委派逻辑,从JDK1.2之后,双亲委派模式已经被引入到类加载体系中,自定义类加载器时不需要在自己写双亲委派的逻辑,因此不鼓励重写loadClass方法,而推荐重写findClass方法。

在Java中,任意一个类都需要由加载它的类加载器和这个类本身一同确定其在java虚拟机中的唯一性,即比较两个类是否相等,只有在这两个类是由同一个类加载器加载的前提之下才有意义,否则,即使这两个类来源于同一个Class类文件,只要加载它的类加载器不相同,那么这两个类必定不相等(这里的相等包括代表类的Class对象的equals()方法、isAssignableFrom()方法、isInstance()方法和instanceof关键字的结果)。例子代码如下:

[java] view plaincopy
  1. package com.test;  
  2.   
  3. public class ClassLoaderTest {  
  4.     public static void main(String[] args)throws Exception{  
  5.         //匿名内部类实现自定义类加载器  
  6.     ClassLoader myClassLoader = new ClassLoader(){  
  7.     protected Class<?> findClass(String name)throws ClassNotFoundException{  
  8.         //获取类文件名  
  9.     String filename = name.substring(name.lastIndexOf(“.”) + 1) + “.class”;  
  10.     InputStream in = getClass().getResourceAsStream(filename);  
  11.     if(in == null){  
  12.     throw RuntimeException(“Could not found class file:” + filename);  
  13. }  
  14. byte[] b = new byte[in.available()];  
  15. return defineClass(name, b, 0, b.length);  
  16. }catch(IOException e){  
  17.     throw new ClassNotFoundException(name);  
  18. }  
  19. };  
  20. Object obj = myClassLoader.loadClass(“com.test.ClassLoaderTest”).newInstance();  
  21. System.out.println(obj.getClass());  
  22. System.out.println(obj instanceof com.test. ClassLoaderTest);  
  23. }  
  24. }  

输出结果如下:

com.test.ClassLoaderTest

false

之所以instanceof会返回false,是因为com.test.ClassLoaderTest类默认使用Application ClassLoader加载,而obj是通过自定义类加载器加载的,类加载不相同,因此不相等。

类加载器双亲委派模型是从JDK1.2以后引入的,并且只是一种推荐的模型,不是强制要求的,因此有一些没有遵循双亲委派模型的特例:

(1).在JDK1.2之前,自定义类加载器都要覆盖loadClass方法去实现加载类的功能,JDK1.2引入双亲委派模型之后,loadClass方法用于委派父类加载器进行类加载,只有父类加载器无法完成类加载请求时才调用自己的findClass方法进行类加载,因此在JDK1.2之前的类加载的loadClass方法没有遵循双亲委派模型,因此在JDK1.2之后,自定义类加载器不推荐覆盖loadClass方法,而只需要覆盖findClass方法即可。

(2).双亲委派模式很好地解决了各个类加载器的基础类统一问题,越基础的类由越上层的类加载器进行加载,但是这个基础类统一有一个不足,当基础类想要调用回下层的用户代码时无法委派子类加载器进行类加载。为了解决这个问题JDK引入了ThreadContext线程上下文,通过线程上下文的setContextClassLoader方法可以设置线程上下文类加载器。

JavaEE只是一个规范,sun公司只给出了接口规范,具体的实现由各个厂商进行实现,因此JNDI,JDBC,JAXB等这些第三方的实现库就可以被JDK的类库所调用。线程上下文类加载器也没有遵循双亲委派模型。

(3).近年来的热码替换,模块热部署等应用要求不用重启java虚拟机就可以实现代码模块的即插即用,催生了OSGi技术,在OSGi中类加载器体系被发展为网状结构。OSGi也没有完全遵循双亲委派模型。

深入理解java虚拟机-读书笔记4-虚拟机类加载机制

虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验、转换解析和初始化,最终形成可以被虚拟机直接使用的java类型,这就是类加载机制。 类加载的时机: 下面5中情况必须立即对类进行”...
  • wsongmao
  • wsongmao
  • 2016年11月30日 20:57
  • 122

【深入理解Java虚拟机】类加载机制

本文内容来源于《深入理解Java虚拟机》一书,非常推荐大家去看一下这本书。本系列其他文章:【深入理解Java虚拟机】Java内存区域模型、对象创建过程、常见OOM【深入理解Java虚拟机】垃圾回收机制...
  • shakespeare001
  • shakespeare001
  • 2016年06月27日 07:53
  • 5430

《深入理解Java虚拟机》学习笔记

本篇是《深入理解Java虚拟机-Java 高级特性与最佳实践》学习笔记,周志明著,Understanding the JVM-Advanced Features and Best Practices,...
  • puma_dong
  • puma_dong
  • 2014年02月23日 11:01
  • 4205

《深入理解java虚拟机》笔记——简析java类文件结构

一直不太搞得明白jvm到底是如何进行类加载的,在看资料的过程中迷迷糊糊,在理解类加载之前,首先看看java的类文件结构到底是怎样的,都包含了哪些内容。  最直接的参考当然是官方文档:The Java®...
  • zhoufenqin
  • zhoufenqin
  • 2016年04月03日 15:31
  • 1864

深入java虚拟机第二版阅读笔记(不断整理中)

1 class文件是如何被调入内存执行的? 加载原理?   深入探讨 Java 类加载器:http://www.ibm.com/developerworks/cn/java/j-lo-classloa...
  • ajian005
  • ajian005
  • 2014年03月05日 11:25
  • 1350

深入理解Java虚拟机笔记(一)

内容主要参考《深入理解Java虚拟机(第2版)》 Java和C++之间有一堵由内存动态分配和垃圾收集技术所围成的“高墙”,墙外面的人想进去,墙里面的人想出来。 一 JVM运行时数据区 yun如下图: ...
  • zoufangyingzi
  • zoufangyingzi
  • 2017年02月20日 18:04
  • 457

【深入Java虚拟机】之四:类加载机制

类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载、验证、准备、解析、初始化、使用和卸载七个阶段。 其中类加载的过程包括了加载、验证、准备、解析、初始化五个阶段。在这...
  • mmc_maodun
  • mmc_maodun
  • 2014年01月08日 00:09
  • 40612

深入理解Java虚拟机之虚拟机执行子系统(读书笔记)

类文件结构 1 Class类文件结构 Class文件的类结构是怎样的? 任何一个Class文件都对应着唯一一个类或接口的定义信息,反过来,类或接口不一定都得定义在文件里(也可通过类加载器直接生成...
  • weixin_36273478
  • weixin_36273478
  • 2017年05月21日 21:52
  • 161

读书笔记——深入理解java虚拟机(一)

深入理解java虚拟机java内存区域与内存溢出异常方法区 Method Area 虚拟机栈 VM stack 本地方法栈 Native method stack 堆 Heap 程序计数器 P...
  • chaoil
  • chaoil
  • 2015年04月01日 20:26
  • 435

读书笔记——深入理解虚拟机

自动内存管理机制 Java与C++之间有一堵由内存分配和垃圾收集所围成的“高强”,墙外面的人想进去,强里面的人却想出来。 java内存区域与内存溢出异常 每个线程私有 程序计数器:可以看做是当前...
  • qq_24145735
  • qq_24145735
  • 2016年08月05日 23:34
  • 467
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:《深入理解java虚拟机》学习笔记6——类加载机制
举报原因:
原因补充:

(最多只允许输入30个字)