深入理解JVM(一)JVM基本原理和类的加载

深入理解JVM基本原理

前言

       我们都知道,JVM一直是java知识里面进阶阶段的重要部分,如果希望在java领域研究的更深入,则JVM则是如论如何也避开不了的话题,本章就带大家去了解一下JVM,讲解一下JVM知识点。

正文   

  1. JVM的编译

          我们都知道java一直宣传的口号是:一次编译,到处运行。 Java的跨平台功能和它的Java虚拟机(简称 JVM)的中介作用是分不开的。所谓跨平台的“平台”指:操作系统。没错,就是我们接触的Linux和Windows等操作系统。Java跨平台,通俗的解释是:用Java编写的程序,既可以在Linux下运行,也可以在Windows下运行。

深入理解JVM(一)JVM基本原理和类的加载_第1张图片

       从上图可以看出java文件经过了一次编译后,java代码编译成java字节码(class文件)。然后再不同平台上使用不同的虚拟机(JVM)解释,解释成机器码,然后执行。由此可见,如果我们要在mac系统上运行,只需要装一个mac 的java虚拟机就可以了。这也就是java的一次编译,到处运行。

  注意

      是不是只有java语言可以使用java虚拟机进行交互呢?答案是否定的

深入理解JVM(一)JVM基本原理和类的加载_第2张图片

如图所示,Jython/Scala/Groovy/JRuby都是可以编译成字节码文件的。所以不是java才可以使用java虚拟机

 2.类的加载机制

    深入理解JVM(一)JVM基本原理和类的加载_第3张图片     类的加载过程总共分为这七步,其中验证、准备、解析也被称为连接。

     接下来仔细了解下这几步是做什么的。

1.加载

         加载简单来说分为三步。

第一步:获取二进制字节流也就是上面的class文件。

第二步:将静态的存储结构转换为方法区中的运行时数据结构。

第三步:生成一个对象放入java堆中,做为对方法区的引用。

2.验证

验证主要是检验如下的几项是否正确 

       class文件的表示(魔数),class文件的版本号,class文件的每个部分是否正确(字段表、方法表等),验证常量池(常量类型、常量类型数据结构是否正确,utf-8是否标准),元数据验证(父类验证,继承验证,final验证),字节码(指令)验证,符号引用验证(是否能根据符号找到对应的字段、表、方法等)

如果一项不对,就会验证失败。

3.准备

      准备阶段为类变量分配内存 和设置类变量初始化。这个过程中,只对static类变量进行内存分配,这个时候只是分配内存,没有进行复制,所有的类变量都是初始化值。如果是final的话,会直接对应到常量池中。会在准备阶段直接赋值。

4.解析

     解析阶段是读符号引用进行解析。将符号引用解析为直接引用(指向目标的指针或者偏移量)。主要涉及到的解析有类,接口,字段,方法等。 

5.初始化

      初始化就是执行方法的过程, 对静态变量,静态代码块进行初始化,对类进行初始化。

6.使用

     使用阶段就是使用这个class。  

7.卸载

     卸载阶段就是不在使用,将class给卸载。

3类加载器

   

 深入理解JVM(一)JVM基本原理和类的加载_第4张图片

如图所示,每当我们类被加载的时候,都会去走类加载器。

      我们自定义加载器有一个父类,就是AppClassLoader,而AppClassLoader也有一个父类加载器ExtClassLoader,ExtClassLoader同样也有一个父类加载器BootstarpClassLoader,BootstarpClassLoader就是最终的加载器了。

    当加载类的时候,会启动类加载器,会通过自己定义的类加载器去找AppClassLoader,然后通过AppClassLoader找到ExtClassLoader,再通过ExtClassLoader找到最终BootstarpClassLoader。可能看到这里,有人人就有疑问了,为什么我有自己的加载器,干嘛还要去找父类的加载器呢?这这种机制叫双亲委任机制,目的就是为了加载累的安全。也就是说我父类加载的不给子类去加载,这样保证最终都是BootstarpClassLoader去加载,可以保证一个类只会加载一次。而你判断两个对象时候一样的时候,最重要的一个条件就是是不是一个加载器去加载的。

  结语

好了,jvm的基本原理就到这里了,以后会完善关于JVM的内存结构以及垃圾回收算法和垃圾回收器相关的知识。 

你可能感兴趣的:(技术,JVM,类加载,类加载器)