java类加载机制详解

类加载基础 ClassLoader

ClassLoader定义:

1、类加载器,负责将class加载到jvm
2、审查每个类应该由谁加载,是一个父优先的等级加载机制
3、将class字节码重新解析成jvm统一要求的对象格式

ClassLoader类结构:

java.lang.ClassLoader

  • defineClass(byte[] b, int off, int len) [java8 中 --> defineClass(String name, byte[] b, int off, int len)]
  • findClass(String name)
  • loadClass(String name)
  • resolveClass(Class c)
  1. findClass方法获取类的字节码
  2. defineClass方法用于将byte字节流解析成jvm能够识别的Class对象(注意该节点并未产生真正的java实例对象,而是类的Class对象)
  3. resolveClass建立连接(即Linking,在调用defineClass方法后,如果想在类被加载到jvm中时就建立连接,调用resolveClass方法)

简单的应用:

java运行时加载指定的类

  • 调用ClassLoader的loadClass方法获取指定类的Class对象
  • this.getClass().getClassLoader().loadClass("class")
    该loadClass存在有重载方法,调用者可以自己决定什么时候解析这个类

ClassLoader是一个抽象方法,在实现自己的ClassLoader是通常可以继承URLClassLoader这个类:
URLClassLoader说明

ClassLoader的等级加载机制:

jvm平台提供三层ClassLoader:
  • Bootstrap ClassLoader:
  1. 主要用于加载jvm自身工作需要的类(仅仅是一个加载工具);
  2. 类的加载过程完全由jvm控制;
  3. 外部不可访问;
  4. 不遵循ClassLoader定义的普遍加载规则;
  5. 没有更高一级的父加载器,也没有子加载器
  • ExtClassLoader:
  1. 该类比较特殊,它是jvm的一部分,但并不是jvm亲自实现的。
  2. * 类似于JDBC这种驱动*
  3. 特定服务目标在:System.getProperty("java.ext.dirs")
  • AppClassLoader:
  1. ExtClassLoader的子类
  2. 用于加载jvm之外的外部类,所有在System.getProperty("java.class.path")即(classPath)下的类都可以被该类加载器加载。
  3. 该类是所有外部加载器(ClassLoader)的父类:
  • 创建自定义的类加载器(ClassLoader)时,无论继承抽象类ClassLoader还是其子类,例如URLClassLoader,他们的父加载器都是AppClassLoader。
  • 因为创建对象的时间,最终都需要调用getSystemClassLoader()作为父加载器,而getSystemClassLoader()方法获得的是AppClassLoader

子父类关系

jvm层级:
  • AppClassLoader继承ExtClassLoader

(一定要注意jvm层级,ExtClassLoader是没有父类的)

java类层级:
  • AppClassLoader,ExtClassLoader属于sun.misc.Launcher类的内部类
  • AppClassLoader,ExtClassLoader都继承了URLClassLoader
  • 在Launcher中,首先创建ExtClassLoader对象,然后ExtClassLoader对象作为父加载器创建AppClassLoader对象,Launcher的getClassLoader()获取的就是AppClassLoader。
  • 在java应用中,如果没有其它的ClassLoader(spring等的自定义ClassLoader),那么除了System.getProperty("java.ext.dirs")目录下的类是由ExtClassLoader加载外,其它的都是由AppClassLoader加载。

jvm加载class文件的两种方式:

  • 隐式加载:不通过代码里调用ClassLoader来加载需要的类,而是通过jvm来自动加载需要的类到内存。
    例如:当类中存在引用或继承时,jvm解析当前类未在内存中发现引用的类,那么jvm就会自动将这些类加载到内存中。
  • 显式加载:与隐式加载相反,在代码中通过调用ClassLoader来加载需要的类。
    例如:调用this.getClass.getClassLoader().loadClass()或者Class.forName(),或者实现自己的ClassLoader的findClass()方法

jvm加载class文件过程:

第一阶段: 找到.class文件,并将字节码加载到内存:class文件 --> findClass

ClassLoader并未定义findClass具体实现,具体的字节码加载到内存中的实现交予其子类处理

  • URLClassLoader中关于findClass()的实现:
/* class文件资源定位 */
private final URLClassPath ucp;

protected Class findClass(final String name) throws ClassNotFoundException {
        final Class result;
        try {
            result = AccessController.doPrivileged(
            new PrivilegedExceptionAction<Class<?>>() {
                    public Class run() throws ClassNotFoundException {
                              String path = name.replace('.', '/').concat(".class");
                              Resource res = ucp.getResource(path, false);
                              if (res != null) {
                                       try {
                                                  return defineClass(name, res);
                                       } catch (IOException e) {
                                                  throw new ClassNotFoundException(name, e);
                                       }
                              } else {
                                        return null;
                              }
                    }
            }, acc);
          } catch (java.security.PrivilegedActionException pae) {
                throw (ClassNotFoundException) pae.getException();
          }
          if (result == null) {
              throw new ClassNotFoundException(name);
          }
          return result;
}

private Class defineClass(String name, Resource res) throws IOException {
            long t0 = System.nanoTime();
            int i = name.lastIndexOf('.');
            URL url = res.getCodeSourceURL();
            if (i != -1) {
                String pkgname = name.substring(0, i);
                // 检查包是否被加载
                Manifest man = res.getManifest();
                definePackageInternal(pkgname, man, url);
            }
            // 读取byt字节流,并创建class对象
            java.nio.ByteBuffer bb = res.getByteBuffer();
            if (bb != null) {
                    // Use (direct) ByteBuffer:
                    CodeSigner[] signers = res.getCodeSigners();
                    CodeSource cs = new CodeSource(url, signers);
                    sun.misc.PerfCounter.getReadClassBytesTime().addElapsedTimeFrom(t0);
                    return defineClass(name, bb, cs);
            } else {
                    byte[] b = res.getBytes();
                    // must read certificates AFTER reading bytes.
                    CodeSigner[] signers = res.getCodeSigners();
                    CodeSource cs = new CodeSource(url, signers);
                    sun.misc.PerfCounter.getReadClassBytesTime().addElapsedTimeFrom(t0);
                    return defineClass(name, b, 0, b.length, cs);
    }
}

URLClassLoader通过URLClassPath类帮助取得要加载的class文件字节流,URLClassPath定义了去哪里找指定的class文件,如果找到class文件,读取其byte字节流,然后调用defineClass()创建对象。

  • URLClassLoader默认类构造器代码如下:
public URLClassLoader(URL[] urls) {
    super();
    // this is to make the stack depth consistent with 1.1
    SecurityManager security = System.getSecurityManager();
    if (security != null) {
            security.checkCreateClassLoader();
    }
    ucp = new URLClassPath(urls);
    this.acc = AccessController.getContext();
}
  • URLClassLoader的创建必须一个URL数组,也就是必须指定ClassLoader默认到那个目录下查找class文件。
  • 该URL数组也是创建URLClassPath必要的条件,URLClassPath通过该URL数组来表示ClassPath路径
  • 创建URLClassPath时会根据URL数组中的路径判断是文件还是jar包,根据路径的不同创建FileLoader或者jarLoader,或者默认的加载器,当jvm调用findClass由这几个加载器来将class文件加载到内存。
  • 实例应用:

文件加载方法的不同:

  • jar包服务、web服务:Tools.class.getClassLoader().getResourceAsStream("url")
  • web服务:Tools.class.getResourceAsStream("url")

第二阶段,Linking阶段:

  1. Class规范验证(字节码验证Verifing)

字节码验证,类装入器对字节码进行检测,以确保格式正确、行为正确。

  1. Class类数据结构分析及相应的内存分配(准备Preparing)

类准备,在这个阶段准备代表每个类中定义的字段、方法和实现接口所必须的数据结构。

  1. 符号表的链接(解析Resolving)

类装入器开始装入类所引用的其他所有类。

第三阶段:

  • 类中静态属性和初始化赋值,以及静态代码块的执行等 --> Class对象

类加载中的常见异常:

  • ClassNotFoundException:
    例如:Class.forName("className") --> ClassNotFoundException
    原因:当前classpath下无该类文件
    解决方法:
    1、添加类文件
    2、类文件存在:this.getClass().getClassLoader().getResource("className").toString()

  • NoClassDefFoundError:
    例如:java -cp example.jar Example
    原因:类没有设置路径
    解决:java -cp example.jar com.ss.Example

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

推荐阅读更多精彩内容

  • 1、classLoader 类加载器,将class文件加载到JVM虚拟机内存中,使得程序可以运行。通常情况下,JV...
    helloWorld_1118阅读 2,197评论 0 2
  • 作者:liuxiaopeng原文地址:http://www.cnblogs.com/paddix/ 一、类加载器 ...
    IT程序狮阅读 3,046评论 2 10
  • 转发:本篇文章已授权微信公众号 guolin_blog (郭霖)独家发布 ClassLoader翻译过来就是类加载...
    尼尔君阅读 527评论 0 1
  • 一:ClassLoader 从JVM结构图中可以看到,类加载器的作用是将Java类文件加载到Java虚拟机。 只有...
    阿菜的博客阅读 1,775评论 0 8
  • ClassLoader翻译过来就是类加载器,普通的java开发者其实用到的不多,但对于某些框架开发者来说却非常常见...
    时待吾阅读 1,058评论 0 1