JVM深刻理解<一>

如下文章来自与:

http://www.jianshu.com/p/fabad9250b1b

1、什么是JVM?

JVM是Java Virtual Machine(Java虚拟机)的缩写,是经过在实际的计算机上仿真模拟各类计算机功能来实现的。由一套字节码指令集、一组寄存器、一个栈、一个垃圾回收堆和一个存储方法域等组成。JVM屏蔽了与操做系统平台相关的信息,使得Java程序只须要生成在Java虚拟机上运行的目标代码(字节码),就可在多种平台上不加修改的运行,这也是Java可以“一次编译,处处运行的”缘由。java

 

2、JRE、JDK和JVM的关系

JRE(Java Runtime Environment, Java运行环境)是Java平台,全部的程序都要在JRE下才可以运行。包括JVM和Java核心类库和支持文件。数组

JDK(Java Development Kit,Java开发工具包)是用来编译、调试Java程序的开发工具包。包括Java工具(javac/java/jdb等)和Java基础的类库(java API )。缓存

JVM(Java Virtual Machine, Java虚拟机)是JRE的一部分。JVM主要工做是解释本身的指令集(即字节码)并映射到本地的CPU指令集和OS的系统调用。Java语言是跨平台运行的,不一样的操做系统会有不一样的JVM映射规则,使之与操做系统无关,完成跨平台性。安全

JDK多线程

JDK(Java Development Kit) 是 Java 语言的软件开发工具包(SDK)。JDK 物理存在,是 programming tools、JRE 和 JVM 的一个集合。函数

 


JRE工具

JRE(Java Runtime Environment)Java 运行时环境,JRE 物理存在,主要由Java API 和 JVM 组成,提供了用于执行 java 应用程序最低要求的环境。性能

 


3、JVM原理

一、JVM的体系结构

 


二、JVM生命周期介绍

Java实例对应一个独立运行的Java程序(进程级别开发工具

1.启动。启动一个Java程序,一个JVM实例就产生。拥有public static void main(String[] args)函数的class能够做为JVM实例运行的起点。优化

2.运行。main()做为程序初始线程的起点,任何其余线程都可由该线程启动。JVM内部有两种线程:守护线程和非守护线程,main()属于非守护线程,守护线程一般由JVM使用,程序能够指定建立的线程为守护线程。

3.消亡。当程序中的全部非守护线程都终止时,JVM才退出;若安全管理器容许,程序也可使用Runtime类或者System.exit()来退出。

JVM执行引擎实例则对应了属于用户运行程序线程它是线程级别的。

三、Java类加载器

 Java加载类的过程

 


1.装载(loading):负责找到二进制字节码并加载至JVM中,JVM经过类名、类所在的包名、ClassLoader完成类的加载。所以,标识一个被加载了的类:类名 + 包名 + ClassLoader实例ID。

2.连接(linking):负责对二进制字节码的格式进行校验、初始化装载类中的静态变量以及解析类中调用的接口。

完成校验后,JVM初始化类中的静态变量,并将其赋值为默认值。

最后对比类中的全部属性、方法进行验证,以确保要调用的属性、方法存在,以及具有访问权限(例如private、public等),不然会形成NoSuchMethodError、NoSuchFieldError等错误信息。

3.初始化(initializing):负责执行类中的静态初始化代码、构造器代码以及静态属性的初始化,如下四种状况初始化过程会被触发。

四、JVM类加载顺序

层级结构

 


1.Booststrap ClassLoader

跟ClassLoader,C++实现,JVM启动时初始化此ClassLoader,并由此完成$JAVA_HONE中jre/lib/rt.jar(Sun JDK的实现)中全部class文件的加载,这个jar中包含了java规范定义的全部接口以及实现。

2.Extension ClassLoader

JVM用此classloader来加载扩展功能的一些jar包

3.System ClassLoader

JVM用此ClassLoader来加载启动参数中指定的ClassPath中的jar包以及目录,在Sun JDK中ClassLoader对应的类名为AppClassLoader。

4.User-Defined ClassLoader

User-Defined ClassLoader是Java开发人员继承ClassLoader抽象类实现的ClassLoader,基于自定义的ClassLoader可用于加载非ClassPath中的jar以及目录。

五、委派模式(Delegation Mode)

 


当JVM加载一个类的时候,下层的加载器会将任务给上一层类加载器,上一层加载检查它的命名空间中是否已经加载这个类,若是已经加载,直接使用这个类。若是没有加载,继续往上委托直到顶部。检查以后,按照相反的顺序进行加载。若是Bootstrap加载器不到这个类,则往下委托,直到找到这个类。一个类能够被不一样的类加载器加载。

可见性限制:下层的加载器可以看到上层加载器中的类,反之则不行,委派只能从下到上

不容许卸载类:类加载器能够加载一个类,但不可以卸载一个类。可是类加载器能够被建立或者删除。

六、JVM执行引擎

类加载器将字节码载入内存后,执行引擎以java字节码为单元,读取java字节码。java字节码机器读不懂,必须将字节码转化为平台相关的机器码。这个过程就是由执行引擎完成的。

 


在执行方法时JVM提供了四种指令来执行

invokestatic:调用类的static方法。

invokevirtual:调用对象实例的方法。

invokeinterface:将属性定义为接口来进行调用。

invokespecial:JVM对于初始化对象(Java构造器的方法为:)以及调用对象实例的私有方法时。

主要的执行计数:

解释,即时执行,自适应优化、芯片级直接执行。

解释属于第一代JVM

即时编译JIT属于第二代JVM

自适应优化(目前sun的HotspotJVM采用这种技术),吸收第一代JVM和第二代JVM的经验,采用二者结合的方式,开始对全部的代码都采用解释执行的方式,并监视代码执行状况,而后对那些常常调用的方法启动一个后台线程,将其编译为本地代码,并进行优化。若方法再也不频繁使用,则取消编译过代码,仍对其进行解释执行。

七、java运行时数据区

 


PC寄存器

用于存储每一个线程下一步将要执行的JVM指令,若该方法为native的,则PC寄存器中不存储任何信息。Java多线程状况下,每一个线程都有一个本身的PC,以便完成不一样线程上下文环境的切换。

JVM栈

JVM栈是线程私有的,每一个线程建立的同时都会建立JVM栈,JVM栈中存放当前线程中局部基本类型的变量(Java中定义的八种基本类型:boolean、char、byte、short、int、long、float、double)、部分的返回结果以及Stack Frame,非基本类型的对象在JVM栈上仅存放一个指向堆的地址。

堆(Heap)

它是JVM用来存储对象实例以及数组值的区域,能够认为Java中全部经过new建立的对象的内存都在此分配,Heap中的对象的内存须要等待GC进行回收。

堆在JVM启动的时候就被建立,堆中储存了各类对象,这些对象被自动管理内存系统(Automatic Storage Management System),也就是常说的“Garbage Collector(垃圾回收器)”管理。这些对象无需、也没法显示地被销毁。

JVM将Heap分为两块:新生代New Generation和旧生代Old Generation

 


堆是JVM中全部线程共享的,所以在其上进行对象内存的分配均须要进行加锁,致使new对象的开销比较大。

Sun Hotspot JVM为了提高对象内存分配的效率,对于全部建立的线程都会分配一块独立的空间TLAB(Thread Local Allocation Buffer),其大小由JVM根据运行的状况计算而得,在TLAB上分配对象时不须要加锁,所以JVM在给线程对象分配内存时会尽可能的在TLAB上分配,在这种状况下JVM中分配对象内存的性能和C基本是同样的,但若是对象过大的话则仍然要直接使用堆空间分配。

TLAB仅做用于新生代的Eden Space,所以在编写Java程序时,一般多个小的对象比大的对象分配起来更加高效。

全部新建立的Object都将会存储在新生代Young Generation中。若是Young Generation的数据在一次或屡次GC后存活下来,那么将被转移到OldGeneration。新的Object老是建立在Eden Space。

方法区域(Method Area)

在Sun JDK中这块区域对应的为PermanetGeneration,又称为持久代。

方法区域存放所加载类的信息(名称、修饰符等)、类中的静态变量、类中定义为final类型的常量、类中的Field信息、类中的方法信息,当开发人员在程序中经过Class对象中的getName,isInstance等方法来获取信息时,这些数据都来源于方法区域,同时方法区域也是全局共享的,在必定条件下它也会被GC,当方法区域须要使用的内存超过其容许的大小时,就会抛出OutOfMemory的错误信息。

运行时常量池(Runtime Constant Pool)

存放的为类中的固定常量信息、方法和Field的引用信息等,其空间从方法区域中分配。

本地方法堆栈(Native Method Stacks)

JVM采用本地方法堆来支持native方法的执行,此区域用于存储每一个native方法调用的状态。

八、JVM垃圾回收

GC的基本原理:将内存中再也不被使用的对象进行回收,GC中用于回收的方法称为收集器,因为GC须要消耗一些资源和时间,Java在对对象生命周期特征进行分析后,按照新生代、旧生代的方式来对对象进行收集,以尽量的缩短GC对应用形成的暂停。

对新生代的对象收集称为minor GC

对旧生代的对象收集称为Full GC

程序中主动调用System.gc()强制执行的GC为Full GC。

不一样的对象引用类型,GC会采用不一样的方法进行回收,JVM对象的引用分为了四种类型

强引用:默认状况下,对象采用的均为强引用(这个对象的实例没有其余对象引用时, GC时才会被回收)

软引用:软引用是Java中提供的一种比较适合于缓存场景的应用(只有内存不够的状况下才会被GC)

弱引用:在GC时必定会被GC回收。

虚引用:虚引用只是用来得知对象是否被GC。

做者:空有一身才华和一张帅气的脸 连接:http://www.jianshu.com/p/fabad9250b1b 來源:简书 著做权归做者全部。商业转载请联系做者得到受权,非商业转载请注明出处。
相关文章
相关标签/搜索