虚拟机创建-启动-类加载-运行-类卸载-虚拟机销毁
上一章节我们大致看了虚拟机的组成,本节将讲述虚拟机的类加载;
- 首先我们需要知道什么是虚拟机的类加载?
Java程序是依赖于Java虚拟机运行的,因此要想让我们的应用程序跑起来,必须先把我们写的程序放进虚拟机里边。应用程序运行时始终获取的都是我们虚拟机内部的数据信息,我们加载的类可以是本地文件,可以是来自于网络,也可以是在运行过程中生成的。因为所有的数据信息都可以以二进制数据的方式保存,因此,我们可以通过各种各样的方式去获取需要加载的类信息。
打个比方:我们想要通过idea去运行我们刚刚写的 hello world,那么我们就必须通过,file -> open的方式把我们的项目导进来。虚拟机加载完成了以后就可以debug 或者run了。(仅仅比喻为什么要加载我们写的类文件,其他差异不予考虑)
- 其次,类加载都做了哪些事?
在上一节我们讲了程序启动的两种方式,Java虚拟机在启动的时候也不知道他应该加载哪些东西。在一开始,除了我们指定的主类之外,虚拟机对我们所编写的程序是一无所知的,所以主类要什么,虚拟机才会去加载什么,这种依赖是会传递的;
举个例子: 想了半天没想出来,挖个坑,以后有灵感了再来填!!!
看下边这段代码,我们想要运行这段程序,就得加载上边三个引入进来的类,
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.ComponentScan;
@ComponentScan({"com.demo.starter","com.example.demo"})
@SpringBootApplication
public class DemoApplication {
public static void main(String[] args) {
SpringApplication application = new SpringApplication(DemoApplication.class);
application.run(args);
}
}
而上边的类又依赖于更多的类,因此,Java虚拟机的类加载就是将我们程序运行时用到的这些相关的class文件加载到内存(虚拟机)中供应用程序去访问。
Java虚拟机会根据我们应用程序的需要去加载相关的class文件,这发生在两种场景之下,一种是我们通过某种方式将类提前加载,比如类加载器在使用某个类之前提前加载。我们可以使用-XX:+TraceClassLoading 参数查看类加载顺序。
- 那么应用程序到底是如何去加载这些类的?
通过我们需要加载的类的全限定名寻找,将目标路径下的文件加载到内存当中。调用本地方法和我们的外部操作系统交互获取类信息。
// 直接加载
Class.forName("com.fc.base.jvm.load.ClassA");
private static native Class<?> forName0(String name, boolean initialize,
ClassLoader loader,
Class<?> caller)
throws ClassNotFoundException;
// 通过类加载器加载
classLoader.loadClass("com.fc.base.jvm.load.ClassA");
private native final Class<?> findLoadedClass0(String name);
- 两种方式加载有什么区别?
public class ClassA {
//静态初始化块
static {
System.out.println("ClassA----我被加载了");
}
}
public class ClassB {
//静态初始化块
static {
System.out.println("ClassB----我被加载了");
}
}
public class Test {
static {
System.out.println("Test----我被加载了");
}
//程序入口:
public static void main(String[] args) throws Exception{
Class.forName("com.fc.base.jvm.load.ClassA");
Class.forName("com.fc.base.jvm.load.ClassB");
}
}
class.forname运行结果如下:
classloader 测试结果:
我们得出结论: 使用Class.forName("");会触发类的初始化,而使用getClassLoader().loadClass("");仅仅只将我们定义的类装载到虚拟机当中。
在程序当中可以通过设置参数,限制使用Class.forName("")时导致的初始化操作。
- 类加载器
类加载器就是加载“类”的工具,在虚拟机当中引入了双亲委派模型的概念,定义了几种不同的类加载器,启动类加载器,扩展类加载器,应用类加载器,用户自定义类加载器。它所实现的目的就是保证我们的同一个“类”只会被同一个类加载器加载,也就是说,即使我们定义了很多类名相同的“类”也可以准确的区分谁是谁。
反之如果我们打破这种规则,就会导致同一个路径下的两个类被不同的类加载器加载,从而出现判断失误。
双亲委派模型的工作原理就是,将加载请求传给父加载器,父加载器加载不到再由子加载器尝试。这样的目的是为了保证加载进虚拟机的类都是唯一的。
举个例子就是:有事找老大,老大干不了了老二干,这样就会形成一个单链,只要有问题,总能定位到固定的某个人身上,不会出现混乱。
Java虚拟机(三)