JVM虚拟机基础知识(JVM位置、类加载生命周期、堆、元空间、jvm常用参数)

JVM是什么

JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的。
引入Java语言虚拟机后,Java语言在不同平台上运行时不需要重新编译。Java语言使用Java虚拟机屏蔽了与具体平台相关的信息,使得Java语言编译程序只需生成在Java虚拟机上运行的目标代码(字节码),就可以在多种平台上不加修改地运行。

JVM的位置

在这里插入图片描述

jvm体系结构

在这里插入图片描述

类加载的生命周期

在这里插入图片描述

JVM类加载机制分为五个部分:加载,验证,准备,解析,初始化

1.加载类文件 从class文件或者jar中,或者从二进制中,以类全名表识存入方法区,供之后的使用。

2.1.验证类文件是否符合jvm的规范,验证一些类的基本信息。格式验证、语法验证、操作验证。
2.2.准备为类的静态变量和常量分配空间和初始值,在堆中分配空间。
2.3.把常量池中的符号引用转为直接引用,可以认为是一些静态绑定的会被解析,动态绑定则只会在运行时进行解析;静态绑定包括一些
final方法(不可以重写),static方法(只会属于当前类),构造器(不会被重写)

3.初始化将一个类中所有被static关键字标识的代码统一执行一遍。
如果执行的是静态变量,那么就会使用用户指定的值覆盖之前在准备阶段设置的初始值;
如果执行的是static代码块,那么在初始化阶段,JVM就会执行static代码块中定义的所有操作。所有变量初始化语句和静态代码块都会在编译时被前端编译器放在收集器里头,存放到一个特殊的方法中,这个特殊的方法就是方法,即类\接口的初始化方法。该方法的作用就是初始化一个类中的变量,使用用户指定的值覆盖之前在准备阶段里设定的值。
任何invoke之类的字节码都无法调用方法,因为该方法只能在类加载的过程中由JVM调用。如果父类还没有被初始化,那么优先对父类初始化,但在方法内部不会显示调用父类的方法,由JVM负责保证一个类的方法执行之前,他的父类方法已经被执行。
JVM必须确保一个类在初始化的过程中,如果是多线程需要同是初始化它,仅仅只能允许其中一个线程对其执行初始化的操作,其余线程必须等待,只有在活动线程执行完对类的初始化操作之后,才会通知正在等待的其他线程。

4.使用
5.卸载

三大主流JVM

1.HotSpot VM (最常用)
2.J9 VM
3.JRockit

程序计数器 (Program Counter Register)

每个线程都有一个程序计数器,是线程私有的,就是一个指针,指向方法区中的方法字节码(用来存储指向像一条指令的地址,也即将要执行的指令代码),在执行引擎读取下一条指令,是一个非常小的内存空间,几乎可以忽略不计。

1.栈:先进后出
2.队列:先进先出(FIFO)
3.一个方法一个栈帧,存储当前线程运行方法所需要的数据,指令,返回地址。线程私有,无GC

堆-heap

存储实例,一个jvm实例只有一个堆内存,线程共享,GC主要就是清理堆。

堆内存模型
在这里插入图片描述
年轻代GC流程

  1. 对象先进入edan区
  2. edan gc后存活对象进入from区,并清空edan
  3. edan满后 from 和edan gc,存活对象进入to区,并清空edan和from
  4. from和to概念调换,持续执行gc(即form现在是to,to为form)
  5. 对象头分代年龄大于15后被已入old
  6. 如果一次回收中,survivor空间不足,超出部分分配到老年代
    注意:survicor中存活的对象大于servivor空间的50%后会直接进入老年代

元空间(方法区)

方法区是被所有线程共享,所有字段和方法字节码,以及一些特殊方法,如构造函数,接口代码也在此定义,简单说,所有订单一的方法的信息都保存在该区域,此区域属于共享区间。
静态变量、常量、类信息(构造方法、接口定义)、运行时的常量池存在方法区中,但是实例变量存储在堆内存中,和方法区无关。
也有人叫该区域为非堆区,但是元空间实际上不在堆中,与堆共享内存,逻辑上可以算在堆中。
jdk1.6及以前:有永久代,字符串常量池和运行时常量池在方法区中
jdk1.7:有永久代,但已逐步"去永久代",字符串常量池移到堆中,运行时常量池还在方法区中(永久代)
jdk1.8:没有永久代了,字符串常量池在堆中,运行时常量池在元空间。

jvm常用参数

-XX:MaxMetaspaceSize 元空间最大值,默认是没有限制的。
-XX:MetaspaceSize 元空间初始空间大小,达到该值就会触发垃圾收集进行类型卸载,同时GC会对该值进行调整
-Xms256m 设置最小堆内存
-Xmx512m 设置最大堆内存
-Xss128k 每个线程堆栈大小,默认256K,减少此数值产生更多线程
-Xmn300m (-XX:NewSize) 设置年轻代内存
-XX:NewRatio 设置年轻代和老年代的比值,如:3 表示年轻代与老年代的比值为1:3
-XX:SurvivorRatio 年轻代中eden区与两个survivor区的比,默认8,即edan:s0:s1=8:1:1
-XX:TargetSurvivorRatio=60 如果Survivor空间的占用超过该设定值,对象在未达到他们的最大年龄
之前就会被提升至老年代。默认值为50,即超过50%的进入老年代
-XX:+PrintGCDetails 打印GC详情 -XX:+PrintGC 与 -verbose:gc 打印GC简略信息
-XX:+PrintCommandLineFlags 打印JVM使用的参数
-XX:MaxTenuringThreshold=3 设置对象在年轻代中经历多少次GC后进入老年代,默认15(0-15之间)
CMS收集器默认为6
-Xloggc:C:\Users\30748\Desktop\JVMTestGC.log 指定GClog存放位置
-XX:+HeapDumpOnOutOfMemoryError 堆溢出时dump hprof格式文件
-XX:HeapDumpPath=C:\Users\30748\java_error_in_idea.hprof 堆溢出文件位置

引用:

部分内容引用地址1:https://www.processon.com/view/5c749debe4b0f9fba6921d15?fromnew=1
部分内容引用地址2:https://www.processon.com/mindmap/5f859a8de401fd06fd870302

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值