JAVA是一门跨平台的语言,这得归功于它的JVM。在实际项目开发中,通常会出现Out of Memery问题,而内存管理是JAVA自己运行管理,要定位问题,前提必须了解JAVA的内存模型,本文将分享JVM内存结构以及一些参数的具体说明
1、定位和导入二进制class文件
2、验证导入类的正确性
3、为类分配初始化内存
4、帮助解析符号引用
执行包在装载类的方法中的指令,也就是方法
虚拟机会在整个计算机内存中开辟一块内存存储JVM需要用到的对象,变量等,运行区数据有分很多小区,分别为:方法区,虚拟机栈,本地方法栈,堆,程序计数器。
垃圾回收器,是负责回收内存中无用的对象,就是这些对象没有任何引用了,它就会被视为垃圾,也就会被删除。
那么类在JVM的执行流程是怎么做的呢?共有三步:加载、链接和初始化。
JVM将java类的二进制形式加载到内存中,并将他缓存在内存中,以便后面使用,如果没有找到指定的类就会抛出异常classNotFound,进程在这里结束。没有错误就继续在Java堆中生成一个代表这个类的java.lang.Class对象,作为方法区域数据的访问入口。
这个阶段做三件事:验证、准备和解析(可选)。
验证是JVM根据java语言和JVM的语义要求检查这个二进制形式。例如,如果篡改经过编译后的类文件,那么这个类文件可能就不能使用了。
准备是指准备要执行的指定的类,准备阶段为变量分配内存并设置静态变量的初始化。在这个阶段分配的仅为类的变量(static修饰的变量),而不包括类的实例变量。对非final的变量,JVM会将其设置成“零值”,而不是其赋值语句的值:
public static int num = 8;
那么在这个阶段,num的值为0,而不是8。 final修饰的类变量将会赋值成真实的值。
解析是检查指定的类是否引用了其他的类/接口,是否能找到和加载其他的类/接口。这些检查将针对被引用的类/接口递归进行,JVM的实施也可以在后面阶段执行解析,即正在执行的代码真正要使用被引用的类/接口的时候。
在这最后一步中,JVM用赋值或者缺省值将静态变量初始化,初始化发生在执行main方法之前。在指定的类初始化前,会先初始化它的父类,此外,在初始化父类时,父类的父类也要这样初始化。这个过程是递归进行的。
简而言之,整个流程是将类存进内存中,检查类的对应调用的类和接口是否可正常使用,再对类进行初始化的过程。
Java代码编译是由Java源码编译器来完成,流程图如下所示:
Java字节码的执行是由JVM执行引擎来完成,流程图如下所示:
主要功能是记录当前线程执行程序的位置,通过改变计数值来确定执行下一条指令。每个线程的创建,都会创建一个程序计数器,并且对于每个线程而言是互相独立的。比如我们在debuger模式下运行,for循环的停止,异常的抛出,都是通过改变该线程对应的计数值来确定下一个执行指令。
来自网络,侵则必删
主要功能是临时存储线程执行到的每个方法需要的参数,其内存空间在编译时就已确定。与程序计数器一样,每创建一个线程,则创建一个虚拟机栈,线程每执行到一个方法,对应的栈里就会创建一个栈帧,栈帧会存储局部变量表、动态链接、操作数和方法出口等信息,执行方法,栈帧入栈,方法执行完,栈帧出栈。
本地方法栈与java虚拟机栈一样,只是记录native方法执行。
堆内存结构
堆内存是存放所有对象实例,也是jvm的GC主要对象。堆内存主要由新生代、生存代、老年代、长久代组成。不同的区域,GC的算法就不一样。新的对象实例创建,会放入到Eden,随着存储对象实例增多,消耗内存接近Eden最大值,则会触发Minor GC,Minor GC之后,则会将活下来的对象实例放入生存区域,生存区域也会被定期扫描,经过多次扫描之后,还存活下来的,则放入老年代,如果老年代内存快消耗完,就触发major GC,也就是full GC动作,将会对整个堆内存进行回收动作。
java中对堆内存设置参数说明:
-Xms:设置堆的最小空间大小。
-Xmx:设置堆的最大空间大小
-Xmn:设置年轻代大小
-XX:NewSize 设置新生代最小空间大小
-XX:MaxNewSize设置新生代最大空间大小方法区
方法区主要存储虚拟机加载类信息、常量、静态变量。方法区也称“永久代”,是所有线程共享的资源。当永久代区域内存消耗解决上限,就会触发FullGC。
-XX:PermSize设置永久代最小空间大小
-XX:MaxPermSize设置永久代最大空间大小
Java虚拟机的生命周期 一个运行中的Java虚拟机有着一个清晰的任务:执行Java程序。
程序开始执行时他才运行,程序结束时他就停止。你在同一台机器上运行三个程序,就会有三个运行中的Java虚拟机。
Java虚拟机总是开始于一个main()方法,这个方法必须是公有、返回void、直接受一个字符串数组。在程序执行时,你必须给Java虚拟机指明这个包换main()方法的类名。 Main()方法是程序的起点,他被执行的线程初始化为程序的初始线程。程序中其他的线程都由他来启动。
Java中的线程分为两种:守护线程(daemon)和普通线程(non-daemon)。守护线程是Java虚拟机自己使用的线程,比如负责垃圾收集的线程就是一个守护线程。当然,你也可 以把自己的程序设置为守护线程。包含Main()方法的初始线程不是守护线程。
只要Java虚拟机中还有普通的线程在执行,Java虚拟机就不会停止。如果有足够的权限,你可以调用exit()方法终止程序。
https://baijiahao.baidu.com/s?id=1631479299299787341&wfr=spider&for=pc
https://baijiahao.baidu.com/s?id=1588663308864677376&wfr=spider&for=pc