在前几年面试Java高级程序员的时候,只要是会一点JVM的基础知识,基本就都可以面试经过了。最近几年,对Java工程师的要求愈来愈严格,对于中级Java工程师来讲,也须要掌握JVM相关的知识了。这不,一名读者出去面试Java中级岗位,就被问及了JVM相关的类的加载、连接和初始化的问题。
本文咱们一块儿讨论Java类的加载、连接和初始化。 Java字节代码的表现形式是字节数组(byte[]),而Java类在JVM中的表现形式是 java.lang.Class类 的对象。一个Java类从字节代码到可以在JVM中被使用,须要通过加载、连接和初始化这三个步骤。这三个步骤中,对开发人员直接可见的是Java类的加 载,经过使用Java类加载器(class loader)能够在运行时刻动态的加载一个Java类;而连接和初始化则是在使用Java类以前会发生的动做。本文会详细介绍Java类的加载、连接和 初始化的过程。java
Java类的加载是由类加载器来完成的。程序员
通常来讲,类加载器分红两类:启动类加载器(bootstrap)和用户自定义的类加载器(user-defined)。面试
二者的区别在于启动类加载器是由JVM的原生代码实现的,而用户自定义的类加载器都继承自Java中的 java.lang.ClassLoader类。在用户自定义类加载器的部分,通常JVM都会提供一些基本实现。应用程序的开发人员也能够根据须要编写本身的类加载器。 JVM中最常使用的是系统类加载器(system),它用来启动 Java应用程序的加载。经过java.lang.ClassLoader的 getSystemClassLoader()方法能够获取到该类加载器对象。bootstrap
类加载器须要完成的最终功能是定义一个Java类,即把Java字节代码转换成JVM中的java.lang.Class类的对象。可是类加载的过程并非这么简单。 数组
Java类加载器有两个比较重要的特征:层次组织结构和代理模式。微信
层次组织结构指的是每一个类加载器都有一个父类加载器,经过 getParent()方法能够获取到。类加载器经过这种父亲-后代的方式组织在一块儿,造成树状层次结构。代理模式则指的是一个类加载器既能够本身完成Java类的定义工做,也能够代理给其它的类加载器来完成。因为代理模式的存在,启动一个类的加载过程的类加载器和最终定义这个类的类加载器可能并非一个。前者称为初始类加载器, 然后者称为定义类加载器。并发
二者的关联在于:一个Java类的定义类加载器是该类所导入的其它Java类的初始类加载器。好比类A经过import导入了类 B,那么由类A的定义类加载器负责启动类B的加载过程。通常的类加载器在尝试本身去加载某个Java类以前,会首先代理给其父类加载器。当父类加载器找不到的时候,才会尝试本身加载。这个逻辑是封装在java.lang.ClassLoader类的 loadClass()方法中的。通常来讲,父类优先的策略就足够好了。在某些状况下,可能须要采起相反的策略,即先尝试本身加载,找不到的时候再代理给父类加载器。这种作法在Java的Web容器中比较常见,也是 Servlet规范推荐的作法。好比,Apache Tomcat为每一个Web应用都提供一个独立的类加载器,使用的就是本身优先加载的策略。 IBM WebSphere Application Server则容许Web应用选择。分布式
类加载器的一个重要用途是在JVM中为相同名称的Java类建立隔离空间。在JVM中,判断两个类是否相同,不只是根据该类的二进制名称 ,还须要根据两个类的定义类加载器。只有二者彻底同样,才认为两个类是相同的。所以,即使是一样的Java字节代码,被两个不一样的类加载器定义以后,所获得的Java类也是不一样的。若是试图在两个类的对象之间进行赋值操做,会抛出 java.lang.ClassCastException。这个特性为一样名称的Java类在JVM中共存创造了条件。在实际的应用中,可能会要求同一名称的Java类的不一样版本在JVM中能够同时存在。经过类加载器就能够知足这种需求。这种技术在 OSGi中获得了普遍的应用微服务
Java类的连接指的是将Java类的二进制代码合并到JVM的运行状态之中的过程。在连接以前,这个类必须被成功加载。类的连接包括验证、准备和解析等几个步骤。验证是用来确保Java类的二进制表示在结构上是彻底正确的。若是验证过程出现错误的话,会抛出 java.lang.VerifyError错误。高并发
准备过程则是建立Java类中的静态域,并将这些域的值设为默认值。准备过程并不会执行代码。在一个Java类中会包含对其它类或接口的形式引用,包括它的父类、所实现的接口、方法的形式参数和返回值的Java类等。解析的过程就是确保这些被引用的类能被正确的找到。解析的过程可能会致使其它的 Java类被加载。不一样的 JVM 实现可能选择不一样的解析策略。
一种作法是在连接的时候,就递归的把全部依赖的形式引用都进行解析。而另外的作法则多是只在一个形式引用真正须要的时候才进行解析。也就是说若是一个 Java 类只是被引用了,可是并无被真正用到,那么这个类有可能就不会被解析。考虑下面的代码:
public class LinkTest { public static void main(String[] args) { ToBeLinked toBeLinked = null; System.out.println("Test link."); } }
类LinkTest 引用了类 ToBeLinked,可是并无真正使用它,只是声明了一个变量,并无建立该类的实例或是访问其中的静态域。
在 Oracle 的 JDK 6 中,若是把编译好的 ToBeLinked 的 Java 字节代码删除以后,再运行 LinkTest,程序不会抛出错误。这是由于 ToBeLinked 类没有被真正用到,而 Oracle 的 JDK 6 所采用的连接策略使得ToBeLinked 类不会被加载,所以也不会发现 ToBeLinked 的 Java 字节代码其实是不存在的。若是把代码改为 ToBeLinked toBeLinked = new ToBeLinked();以后,再按照相同的方法运行,就会抛出异常了。由于这个时候 ToBeLinked 这个类被真正使用到了,会须要加载这个类。
当一个 Java 类第一次被真正使用到的时候,JVM 会进行该类的初始化操做。初始化过程的主要操做是执行静态代码块和初始化静态域。在一个类被初始化以前,它的直接父类也须要被初始化。可是,一个接口的初始化,不会引发其父接口的初始化。在初始化的时候,会按照源代码中从上到下的顺序依次执行静态代码块和初始化静态域。考虑下面的代码:
public class StaticTest { public static int X = 10; public static void main(String[] args) { System.out.println(Y); //输出60 } static { X = 30; } public static int Y = X * 2; }
在上面的代码中,在初始化的时候,静态域的初始化和静态代码块的执行会从上到下依次执行。所以变量 X 的值首先初始化成 10,后来又被赋值成 30;而变量 Y 的值则被初始化成 60。
Java 类和接口的初始化只有在特定的时机才会发生,这些时机包括:
MyClass obj = new MyClass()
MyClass.sayHello()
MyClass.value = 10
int value = MyClass.value
assert true;
经过 Java 反射 API 也可能形成类和接口的初始化。须要注意的是,当访问一个 Java类或接口中的静态域的时候,只有真正声明这个域的类或接口才会被初始化。以下面的代码所示。
package io.mykit.binghe.test; class B { static int value = 100; static { System.out.println("Class B is initialized."); // 输出 } } class A extends B { static { System.out.println("Class A is initialized."); // 不会输出 } } public class InitTest { public static void main(String[] args) { System.out.println(A.value); // 输出100 } }
在上述代码中,类 InitTest 经过 A.value 引用了类 B 中声明的静态域 value。因为 value是在类 B 中声明的,只有类 B 会被初始化,而类 A 则不会被初始化。
在 Java 应用开发过程当中,可能会须要建立应用本身的类加载器。典型的场景包括实现特定的 Java 字节代码查找方式、对字节代码进行加密/解密以及实现同名 Java 类的隔离等 。建立 本身的 类加载 器并不 是 一件复杂 的事情 ,只须要继承自java.lang.ClassLoader 类并覆写对应的方法便可。 java.lang.ClassLoader 中提供的方法有很多,下面介绍几个建立类加载器时须要考虑的:
这里比较 容易混淆的是 findClass()方法和 loadClass()方法的做用。前面提到过,在Java 类的连接过程当中,会须要对 Java 类进行解析,而解析可能会致使当前 Java 类所引用的其它 Java 类被加载。在这个时候,JVM 就是经过调用当前类的定义类加载器的 loadClass()方法来加载其它类的。 findClass()方法则是应用建立的类加载器的扩展点。应用本身的类加载器应该覆写 findClass()方法来添加自定义的类加载逻辑。loadClass()方法的默认实现会负责调用 findClass()方法。前面提到,类加载器的代理模式默认使用的是父类优先的策略。这个策略的实现是封装在 loadClass()方法中的。若是但愿修改此策略,就须要覆写 loadClass()方法。
下面的代码给出了自定义的类加载的常见实现模式
public class MyClassLoader extends ClassLoader { protected Class<?> findClass(String name) throws ClassNotFoundException { byte[] b = null; //查找或生成Java类的字节代码 return defineClass(name, b, 0, b.length); } }
好了,今天就到这儿吧,我是冰河,咱们下期见!!
微信搜一搜【冰河技术】微信公众号,关注这个有深度的程序员,天天阅读超硬核技术干货,公众号内回复【PDF】有我准备的一线大厂面试资料和我原创的超硬核PDF技术文档,以及我为你们精心准备的多套简历模板(不断更新中),但愿你们都能找到心仪的工做,学习是一条时而郁郁寡欢,时而开怀大笑的路,加油。若是你经过努力成功进入到了心仪的公司,必定不要懈怠放松,职场成长和新技术学习同样,不进则退。若是有幸咱们江湖再见!
另外,我开源的各个PDF,后续我都会持续更新和维护,感谢你们长期以来对冰河的支持!!
若是你以为冰河写的还不错,请微信搜索并关注「 冰河技术 」微信公众号,跟冰河学习高并发、分布式、微服务、大数据、互联网和云原生技术,「 冰河技术 」微信公众号更新了大量技术专题,每一篇技术文章干货满满!很多读者已经经过阅读「 冰河技术 」微信公众号文章,吊打面试官,成功跳槽到大厂;也有很多读者实现了技术上的飞跃,成为公司的技术骨干!若是你也想像他们同样提高本身的能力,实现技术能力的飞跃,进大厂,升职加薪,那就关注「 冰河技术 」微信公众号吧,天天更新超硬核技术干货,让你对如何提高技术能力再也不迷茫!