深入理解JVM基本原理
前言
我们都知道,JVM一直是java知识里面进阶阶段的重要部分,如果希望在java领域研究的更深入,则JVM则是如论如何也避开不了的话题,本章就带大家去了解一下JVM,讲解一下JVM知识点。
正文
-
JVM的编译
我们都知道java一直宣传的口号是:一次编译,到处运行。 Java的跨平台功能和它的Java虚拟机(简称 JVM)的中介作用是分不开的。所谓跨平台的“平台”指:操作系统。没错,就是我们接触的Linux和Windows等操作系统。Java跨平台,通俗的解释是:用Java编写的程序,既可以在Linux下运行,也可以在Windows下运行。
从上图可以看出java文件经过了一次编译后,java代码编译成java字节码(class文件)。然后再不同平台上使用不同的虚拟机(JVM)解释,解释成机器码,然后执行。由此可见,如果我们要在mac系统上运行,只需要装一个mac 的java虚拟机就可以了。这也就是java的一次编译,到处运行。
注意
是不是只有java语言可以使用java虚拟机进行交互呢?答案是否定的
如图所示,Jython/Scala/Groovy/JRuby都是可以编译成字节码文件的。所以不是java才可以使用java虚拟机
2.类的加载机制
类的加载过程总共分为这七步,其中验证、准备、解析也被称为连接。
接下来仔细了解下这几步是做什么的。
1.加载
加载简单来说分为三步。
第一步:获取二进制字节流也就是上面的class文件。
第二步:将静态的存储结构转换为方法区中的运行时数据结构。
第三步:生成一个对象放入java堆中,做为对方法区的引用。
2.验证
验证主要是检验如下的几项是否正确
class文件的表示(魔数),class文件的版本号,class文件的每个部分是否正确(字段表、方法表等),验证常量池(常量类型、常量类型数据结构是否正确,utf-8是否标准),元数据验证(父类验证,继承验证,final验证),字节码(指令)验证,符号引用验证(是否能根据符号找到对应的字段、表、方法等)
如果一项不对,就会验证失败。
3.准备
准备阶段为类变量分配内存 和设置类变量初始化。这个过程中,只对static类变量进行内存分配,这个时候只是分配内存,没有进行复制,所有的类变量都是初始化值。如果是final的话,会直接对应到常量池中。会在准备阶段直接赋值。
4.解析
解析阶段是读符号引用进行解析。将符号引用解析为直接引用(指向目标的指针或者偏移量)。主要涉及到的解析有类,接口,字段,方法等。
5.初始化
初始化就是执行<clinte><linte>方法的过程, <clinte>对静态变量,静态代码块进行初始化,<linte>对类进行初始化。
6.使用
使用阶段就是使用这个class。
7.卸载
卸载阶段就是不在使用,将class给卸载。
3类加载器
如图所示,每当我们类被加载的时候,都会去走类加载器。
我们自定义加载器有一个父类,就是AppClassLoader,而AppClassLoader也有一个父类加载器ExtClassLoader,ExtClassLoader同样也有一个父类加载器BootstarpClassLoader,BootstarpClassLoader就是最终的加载器了。
当加载类的时候,会启动类加载器,会通过自己定义的类加载器去找AppClassLoader,然后通过AppClassLoader找到ExtClassLoader,再通过ExtClassLoader找到最终BootstarpClassLoader。可能看到这里,有人人就有疑问了,为什么我有自己的加载器,干嘛还要去找父类的加载器呢?这这种机制叫双亲委任机制,目的就是为了加载累的安全。也就是说我父类加载的不给子类去加载,这样保证最终都是BootstarpClassLoader去加载,可以保证一个类只会加载一次。而你判断两个对象时候一样的时候,最重要的一个条件就是是不是一个加载器去加载的。
结语
好了,jvm的基本原理就到这里了,以后会完善关于JVM的内存结构以及垃圾回收算法和垃圾回收器相关的知识。