JVM类加载机制剖析

一、何为类加载器

我们编写的.java文件经过编译器编译之后,生成.class文件,即字节码文件,类加载器就是负责加载字节码文件到JVM中,并将字节码转换成为java.lang.class类的实例,这个实例便是我们编写的类,通过class实例的newInstance方法,便可以得到java类的对象。

类加载器是类加载过程中的关键角色,他存在于「类加载Class load」过程的「加载」阶段中,在这个阶段,JVM虚拟机完成了三件事情:

  1. 通过一个类的全限定名(包名称+类名称)获取定义此类的二进制字节流(类的权限定名可以映射到文件系统中的文件路径);
  2. 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构;
  3. 在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口;

java.lang.ClassLoader

  1. loadClass(String className): 加载className类,返回java.lang.class类的实例,异常则抛出ClassNotFoundException
  2. defineClass(String name, byte[] b, int off, int len): 加载字节码,返回class类的实例,异常则抛出NoClassDefFoundError

二、类加载器的体系结构

1. 启动类加载器「Bootstrap ClassLoader」

处于最顶端的类加载器,主要负责JAVA_HOME/jre/lib目录下的核心jar或者由-Xbootclasspath选项指定的jar包的装入工作。深入分析下Launcher的源码,发现Bootstrap ClassLoader其实加载的是System.getProperty("sun.boot.class.path")定义下的类包路径。

查看JVM启动后Bootstrap ClassLoader具体加载了哪些jar:

URL[] bootUrls = sun.misc.Launcher.getBootstrapClassPath().getURLs();
for (URL url : bootUrls) {
    System.out.println(url.toExternalForm());
}

Bootstrap ClassLoader是由C++编写的并且内嵌于JVM中,该加载器是无法被java程序直接引用的。比如,java.util.ArrayList类处于rt.jar包下,该包是由Bootstrap ClassLoader负责加载,所以下面这段代码打印出来就是null了。

ArrayList list = new ArrayList();
System.out.println("list的类加载器为:"+list.getClass().getClassLoader());

2. 扩展类加载器「Extension ClassLoader」

扩展类加载器是由sun.misc.Launcher$ExtClassLoader实现,顾名思义这个类加载器主要负责加载JAVA_HOME\lib\ext目录中或者被java.ext.dirs系统变量定义的路径下的所有类库。

3. 应用程序类加载器「App ClassLoader」

应用程序类加载器是由sun.misc.Launcher$AppClassLoader实现,通过源码发现,该类加载器负责加载System.getProperty("java.class.path")也就是classpath下的类库。该类加载器又可以称为系统类加载器,在用户没有明确指定类加载器的情况下,系统默认使用AppClassLoader加载类。

4. 自定义类加载器「Custom ClassLoader」

自定义类加载器是提供给用户自定义「加载哪里的类」而产生的,当初虚拟机在定义「通过一个类的全限定名(包名称+类名称)获取定义此类的二进制字节流」并没有把获取方式限定死,提供了灵活的方式给用户使用,被加载的类可以来自于数据库、可以来自本地文件、可以来自云存储介质等等,用户所需要的就是自定义类加载器并且继承ClassLoader,最后重写「findClass」方法,ClassLoader为我们提供了defineClass方法可以方便的加载源码的二进制字节流。

/*
 * for example, an application could create a network class loader to
 * download class files from a server.  Sample code might look like:
 */

ClassLoader loader = new NetworkClassLoader(host,port);
Object main = loader.loadClass("Main", true).newInstance();

/*
 *The network class loader subclass must define the methods {@link
 * #findClass <tt>findClass</tt>} and <tt>loadClassData</tt> to load a class
 * from the network.  Once it has downloaded the bytes that make up the class,
 * it should use the method {@link #defineClass <tt>defineClass</tt>} to
 * create a class instance.  A sample implementation is:
 */

class NetworkClassLoader extends ClassLoader {
 String host;
 int port;

 public Class findClass(String name) {
    byte[] b = loadClassData(name);
    return defineClass(name, b, 0, b.length);
 }
 private byte[] loadClassData(String name) {
    // load the class data from the connection
 }
}
classloader.png

三、双亲委派模型

在同一个JVM中,一个类加载器和一个类的全限定名共同唯一决定一个类Class的实例。也就是说判定两个类相等法则:类的全名(包名+类名)相同+类加载器相同。

类加载器在加载类的过程中,会先代理给它的父加载器,以此类推。这样的好处是能够保证Java核心库的类型安全,例如java.lang.String类,如果不存在代理模式,则不同的类加载,根据判定两个类相等的法则,会导致存在不同版本的String类,会导致不兼容问题。

核心代码:

protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            //  [首先,检查该类是否被当前类加载器加载]
            Class c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
            // [调用父类加载器的loadClass方法,实现了自底向上的检查类是否被加载的功能]
                        c = parent.loadClass(name, false);
                    } else {
             // [父类加载器为null,也就是去调用BootClassLoader加载]
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }
                if (c == null) {
                    long t1 = System.nanoTime();
                    // [调用当前类加载器findClass方法实现了的自顶向下的类加载功能:ExtClassLoader.findClass(name) -> AppClassLoader.findClass(name) -> CustomClassLoader.findClass(name)]
                    c = findClass(name);
                    // this is the defining class loader; record the stats
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

四、类加载过程

在前面介绍类加载器的代理模式的时候,提到过类加载器会首先代理给其它类加载器来尝试加载某个类。这就意味着真正完成类的加载工作的类加载器和启动这个加载过程的类加载器,有可能不是同一个。真正完成类的加载工作是通过调用 defineClass来实现的;而启动类的加载过程是通过调用 loadClass来实现的。前者称为一个类的定义加载器(defining loader),后者称为初始加载器(initiating loader)。在 Java 虚拟机判断两个类是否相同的时候,使用的是类的定义加载器。也就是说,哪个类加载器启动类的加载过程并不重要,重要的是最终定义这个类的加载器。两种类加载器的关联之处在于:一个类的定义加载器是它引用的其它类的初始加载器。如类 com.example.Outer引用了类com.example.Inner,则由类 com.example.Outer的定义加载器负责启动类 com.example.Inner的加载过程。

类加载器在成功加载某个类之后,会把得到的 java.lang.Class类的实例缓存起来。下次再请求加载该类的时候,类加载器会直接使用缓存的类的实例,而不会尝试再次加载。也就是说,对于一个类加载器实例来说,相同全名的类只加载一次,即 loadClass方法不会被重复调用。

参考资料:
http://www.ibm.com/developerworks/cn/java/j-lo-classloader/

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容