Java 类的加载,链接,初始化

更多 Java 虚拟机方面的文章,请参见文集《Java 虚拟机》


一个类 Person 从代码到使用:

  • 编译器负责将 Person.java 源文件编译为 Person.class 字节码文件
  • 类加载器 Class Loader 负责将 Person.class 字节码 (表现形式为字节数组 byte[])转换为 JVM 中的 Class<Person> 对象
  • 随后 JVM 再利用 Class<Person> 对象 实例化为 Person 对象

1. 类的加载

1.1 类加载器 Class Loader

作用:

  • 将 .class 文件中的字节码转换为 JVM 中的 Class 对象(不是 Class 的实例)
  • 为 JVM 中相同名称的类创建隔离空间。使得同一名称不同版本的两个 Java 类可以在 JVM 中同时存在,例如 OSGI。
    在 JVM 中判断两个类是否相同:类的二进制名称相同 并且 类加载器相同

类加载器 Class Loader 具有层次组织结构,即每个类加载器都有一个父类加载器,通过 getParent() 可以获得父类加载器。

类加载器 Class Loader 使用代理模式,每个类加载器即可以自己完成 Java 类的定义工作,也可以代理给其他的类加载器来完成。

  • 初始类加载器:启动一个类的加载过程
  • 定义类加载器:负责最终定义这个类。
    例如在下面的代码中,A 的定义类加载器负责启动 B 的加载过程
class A {
    private B b;
}

1.2 类加载器 Class Loader 的加载策略

  • 类加载器在尝试自己去加载某个类之前,会首先代理给父类加载器。当父类加载器在 class path 中找不到对应的 .class 字节码文件时,才会尝试自己加载。
    一般的 Java 应用使用该策略。从 ClassLoaderloadClass() 方法中可以看出 c = parent.loadClass(name, false);
protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // First, check if the class has already been loaded
            Class<?> c = findLoadedClass(name);
            if (c == null) {
                long t0 = System.nanoTime();
                try {
                    if (parent != null) {
                        c = parent.loadClass(name, false);
                    } else {
                        c = findBootstrapClassOrNull(name);
                    }
                } catch (ClassNotFoundException e) {
                    // ClassNotFoundException thrown if class not found
                    // from the non-null parent class loader
                }

                if (c == null) {
                    // If still not found, then invoke findClass in order
                    // to find the class.
                    long t1 = System.nanoTime();
                    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;
        }
    }
  • 相反策略,类加载器首先尝试自己去加载某个类,当其在 class path 中找不到对应的 .class 字节码文件时,再代理给父类加载器。
    该策略在 Java Web 容器中比较常见。Apache Tomcat 为每个 Application 提供一个独立的类加载器 WebappClassLoader,使得 Application 自己的类的优先级高于 Web 容器提供的类,因此不同的 Application 可以使用不同版本的库。

1.3 JVM 自带的 Class Loader

  • SystemClassLoader:C++编写,加载核心库 java.*
  • ExtClassLoader:Java编写,加载扩展库 javax.*
  • AppClassLoader:Java编写,加载程序所在目录

通过 Thread.currentThread().getContextClassLoader() 获得当前类加载器

public static void main(String[] args) {
    ClassLoader cl = Thread.currentThread().getContextClassLoader();
    System.out.println(cl.toString());

    try {
        Class c = cl.loadClass("jvm.Person");
        System.out.println(c.getClassLoader());
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
}

输出:

sun.misc.LauncherAppClassLoader@75b84c92 sun.misc.LauncherAppClassLoader@75b84c92

1.4 自定义类加载器 Class Loader

继承父类 ClassLoaderClassLoader 中包含如下方法:

  • final Class<?> defineClass(String name, byte[] b, int off, int len)
    • 将字节码数组转换为 Class<?> 对象
    • 该方法不能被 override
  • final Class<?> findLoadedClass(String name)
    • 查找已经加载过的 Class<?> 对象,即 Java 类
    • 一个类加载器不会重复加载同一个类
    • 该方法不能被 override
  • Class<?> findClass(String name)
    • 根据名称查找并加载 Java 类
    • 该方法需要被 override
  • Class<?> loadClass(String name)
    • 根据名称加载 Java 类
    • 该方法不能被 override

例如我们可以自定义一个类加载器负责从网络中获取字节码,并转化为 Class 对象。

class MyClassLoader extends ClassLoader {
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        byte[] bytes = new byte[1024];
        // 从 网络中根据 name 读取 字节数组

        // 将字节码数组转换为 Class<?> 对象
        return defineClass(name, bytes, 0, bytes.length);
    }
}

1.5 显示加载 VS 隐式加载

  • 显示加载:
    • 通过 Class c = Class.forName("Student");
    • 通过 ClassLoader 的 loadClass() 方法,例如:
ClassLoader cl = Thread.currentThread().getContextClassLoader();
Class c = cl.loadClass("Student");

2. 类的链接

类的链接:将 Java 类的二进制代码合并到 JVM 的运行状态之中的过程。

包括三个步骤:

  • 验证:确保 Java 类的二进制表示在结构上是合理的
  • 准备:创建静态域并赋值
  • 解析:确保当前类引用的其他类被正确地找到,该过程可能会触发其他类被加载。

关于解析,不同的 JVM 有不同的解析策略,例如:

public class A {
  public void main(String args[]) {
    B b = null;
  }
}
  • 策略1:链接 A 的时候发现引用了 B,因此加载 B
  • 策略2:链接 A 的时候发现引用了 B,但是 B 没有被使用,因此不加载 B。在真正使用 B 时才加载 B,例如 b = new B();

3. 类的初始化

类的初始化:当 Java 类第一次被真正使用的时候,JVM 会负责初始化该类。包括:

  • 执行静态代码块
  • 初始化静态域

注意:是类的初始化,不是对象的初始化。

例如:下面的代码不会初始化类 A,因为 A 没有真正被使用。

public static void main(String[] args) {
    A a;
}

static class A {
    static int i = 10;
    static {
        System.out.println("Init class A");
    }
}

例如:下面的代码会初始化类 A,因为 A 真正被使用,输出 Init class A

public static void main(String[] args) {
    int i = A.i;
}

static class A {
    static int i = 10;
    static {
        System.out.println("Init class A");
    }
}

引用:
Java深度历险(二)——Java类的加载、链接和初始化

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

推荐阅读更多精彩内容