JAVA类隔离机制
目录
类加载概述
什么是类加载
- 我们写的java代码(.java文件)经过编译器编译后,生成字节码文件(.class文件),class文件中包含了JVM虚拟机指令集和符号表等信息。虚拟机把class文件的数据加载到内存,并对数据做一些校验、转换解析以及初始化,最终形成可以被jvm直接使用的java类型,就是虚拟机的类加载机制。
类加载的好处
在java程序中,类的加载、链接、初始化都是在运行期完成的,这种方式会稍微增加一些性能损耗,但是提高了java程序的灵活度。比如
- 面向接口的程序,可以再运行时在指定其具体的实现类。(举例:bcp的校验脚本编写,doom的自定义子调用处理等)
- 通过自定义classLoader,可以从网络或其他地方加载特定的class文件。(所以class文件不仅仅指存在本地磁盘的class文件,可以是任何符合jvm虚拟机规范的二进制流。)
类加载流程
类加载时机
一个java类从被加载到虚拟机内存,到卸载出内存,其生命周期包含7个阶段:
什么情况下开始类加载阶段的第一个阶段 加载? java虚拟机并没有进行强制约束,但是以下5中情况,虚拟机规范则是严格规定了必须立即对类进行初始化:
- 1、使用new关键字实例化对象的时候、读取或设置一个类的静态字段(被final修饰、已在编译期把结果放入常量池的静态字段除外)的时候,以及调用类的静态方法的时候。
- 2、使用java.lang.reflect包的方法对类进行反射调用的时候。
- 3、初始化一个类的子类的时候。若其父类未被初始化,则先触发父类的初始化。
- 4、虚拟机执行启动时,被标明为启动类的类(包含main的类)。
- 5、使用jdk1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后解析结果是REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。(关于方法句柄,其功能和反射类似,有点是更灵活简洁,可以参考简单了解下:java7新特性之方法句柄MethodHandle使用_零度的博客专栏-CSDN博客_java methodhandle)
除主动引用(上述5种场景)之外,其他所有引用类的方式都不会触发初始化,这些其他类的引用方式称为被动使用。
比如:
- 1、通过子类引用父类的静态字段,不会导致子类初始化。
- 2、通过数组定义来引用类,不会触发此类的初始化。(举例:ClassA[] aList= new ClassA[10])
- 3、常量在编译阶段会存入调用类的常量池中,本质上没有直接引用到定义常量的类,因此不会触发。
- 4、通过类名获取Class对象,不会触发类的初始化。
- 5、通过Class.forName加载指定类时,如果指定参数initialize(这个参数告诉虚拟机,要不要对类进行初始化)为false时,也不会触发类的初始化。
- 6、通过ClassLoader默认的loadClass方法,也不会触发初始化。
加载顺序
JVM自带有三个类加载器,jvm的类加载顺序也是以下三个顺序来的:
- Bootstrap ClassLoader
-
- 最顶层的加载类,主要加载核心类库,%JRE_HOME%\lib下的rt.jar、resources.jar、charsets.jar和class等。可以通过启动jvm时指定-Xbootclasspath和路径来改变Bootstrap ClassLoader的加载目录。比如
java -Xbootclasspath/a:path
将指定的文件追加到默认的bootstrap路径中。
- 最顶层的加载类,主要加载核心类库,%JRE_HOME%\lib下的rt.jar、resources.jar、charsets.jar和class等。可以通过启动jvm时指定-Xbootclasspath和路径来改变Bootstrap ClassLoader的加载目录。比如
- Extention ClassLoader
-
- 扩展的类加载器,加载目录%JRE_HOME%\lib\ext目录下的jar包和class文件。可以通过
-D java.ext.dirs
指定的加载目录。
- 扩展的类加载器,加载目录%JRE_HOME%\lib\ext目录下的jar包和class文件。可以通过
- Appclass Loader也称为SystemAppClass
-
- 加载当前应用classpath下所有类。
加载顺序为什么是这样,直接看源码
Luancher源码
sun.misc.Launcher,是java虚拟机的入口,下面截了部分Launcher的代码:
// This class is used by the system to launch the main application. public class Launcher { private static Launcher launcher = new Launcher(); private static String bootClassPath = System.getProperty("sun.boot.class.path"); private ClassLoader loader; public static Launcher getLauncher() { return launcher; } public Launcher() { Launcher.ExtClassLoader var1; try { //初始化ExtClassLoader var1 = Launcher.ExtClassLoader.getExtClassLoader(); } catch (IOException var10) { throw new InternalError("Could not create extension class loader"); } try { //初始化APPCLassLoader this.loader = Launcher.AppClassLoader.getAppClassLoader(var1); } catch (IOException var9) { throw new InternalError("Could not create app