jvm总结学习

四种加载器

1.启动类加载器

2.拓展类加载器

3.应用程序加载器

4.自定义加载器

沙箱机制

就是为了保证安全,增加的一些权限。

native方法区(静态变量,常量,类信息(构造方法,接口定义),运行时的常量池)

1.凡是带了native的关键字,说明java的作用范围达不到了,会去底层调用C语言的库。

2.会进入本地方法栈

3.调用本地方法本地接口 JNI

4.JNI作用:会拓展java的使用融合不同的编程语言为Java使用! 最初是C C++

java诞生的时候,C C++横行 ,想要立足,必须要有调用 C C++的方法

5.它在内存区专门开辟了一个空间 Native Method Stack 来标记native方法,

6.在最终执行的时候,加载本地方法库中方法通过JNI

堆(重点)

一个jvm只有一个堆,堆的大小可以调节, 一般 类,方法,常量,变量 ,保存我们所有引用类型的真实对象;

堆内存还分为3个区域;

新生区

老年区

永久区 用来存在JDK字自身携带的Class对象。interface元数据,存储的是java运行时的一些环境或者类信息。

GC垃圾回收,主要是在新生区和老年区

在JDK1.8之后 永久区改为元空间 方法区在元空间中 常量池在方法区中

元空间只在逻辑上存在,物理上是不存在的。

如果出现OOM 堆内存溢出 怎么解决:

1.尝试扩大内存看结果,

2.分析内存,看下哪个地方出现了问题(死循环,递归调用之类的)

GC

什么是GC root?

JVM在垃圾回收的时候,需要找到"垃圾"对象,也就是没有被引用的对象,但是直接找垃圾对象是比较耗时间的,所以反过来找,先找非垃圾对象,也就是正常的对象,那么就要从某些"根"去找,根据这些根的引用路径找到正常的对象,而这些根有个特征,就是它会引用其他的对象,而不会被其他的对象引用,例如:栈中的本地变量,方法区的静态变量,本地方法栈的变量,正在运行的线程等都可以作为GC root;

jvm在进行垃圾回收的时候,并不是对这三个区域统一回收。大部分的时候,回收的都是新生代

三个区域进行垃圾回收

新生代

幸存区

老年区

GC 分为两种 轻GC(普通的GC),重GC(全局的GC)

GC题目:

JVM内存模型和分区~详细到每个区都放什么?(看上面的图片)

堆里面的分区有哪些? Eden,form,to,old,说说他们的特点!

GC的算法? 标记-清除法,标记整理,复制算法,引用计数法(不常用) 这些怎么用的?(下面有详细的工作原理介绍)

轻GC和重GC分别在什么时候发生?

轻GC是在新生区、存活区进行发生的。

重GC是在新生区、存活区、老年区进行发生的。

复制算法:

新生区 :幸存区(from):幸存区(to)=8:1:1 原理就是在新生区产生的内存放在幸存区from中,然后两个幸存区 ,谁的空间是空的 谁就是to区 ,在内存不断的生成中,两个幸存区来回交换,即:最终保留一个to区 也就是空的,另外一个from存放的就是活下来的数据。

好处:没有内存的碎片

坏处:浪费了内存的空间:多了一半空间永远是空的to区

试用的场景:对象存活度较低的时候:新生区

标记-清除算法:

好处:不需要额外的空间。

坏处:两次扫描,严重浪费时间,会产生内存碎片。

试用的场景:对象存活度较低的时候:新生区

JVM1.7和1.8java虚拟机发生了哪些变化?

1.7中存在永久代,1.8改为元空间,元空间所占的内存不是虚拟机内部,而是本地内存空间,不管是永久代还是元空间,他们都是方法的具体实现,之所以元空间所占的内存改为本地内存,官方的说法是为了和JRockit统一,不过还有其他的一些原因,比如:方法区所存储的类信息通常是比较难确定的,所以对于方法区的大小是很难确定的,太小了容易出现方法区的溢出,太大了又会占用虚拟机更多的 空间,而转移到本地,则不会影响虚拟机所占的内存。

总结

内存效率:复制算法>标记清除算法>标记整理(压缩)算法 说白了 就是比较 时间复杂度

内存整齐度:复制算法=标记整理(压缩)算法 >标记清除算法 说白了 就是比较 内存连不连续

内存利用率:标记清除算法 =标记整理(压缩)算法 >复制算法

没有最优的算法,只有最合适的算法------>GC:分代收集算法

年轻代:

存活率低

复制算法!

老年代:

区域大:存活率

标记清除(内存碎片不是很多)+标记压缩(整理) 混合实现

判断 对象是否存活

引用计数法---------每当有一个地方引用它时,计数器值就加1,当索引失效时,计数器就减1;任何时刻计数器为0的对象就不可能再被使用。

可达性分析算法---------这个算法的基本思路就是通过一系列的称为"GC Roots"的对象作为起始点,从这些节点开始向下搜索,搜索走过的路径称为引用链,当一个对象到GC Roots没有任何引用链相连时(按照图说的话,就是从GC Roots到这个对象不可达),则证明这个对象是不可用的,如下图,obj5、obj6、obj7、虽然互相有关联,但是都是GC Roots是不可达的,所以他们将被判定为可回收的对象。

垃圾回收算法

标记-清除算法(Mark-Sweep)---最基础的算法,分为两个阶段,标注和清除,标注阶段标出所有需要清除的对象,清除阶段会回收被标记对象所占用的空间

复制算法---为了解决Mark-Sweep算法内存碎片化的缺陷而被提出的算法。将内存容量将内存划分为等大小的两块。每次只使用其中一块,当这一块内存满后将尚存活的对象复制到另一块上去,把已经使用的内存清掉,如图

标记-整理算法(Mark-Compact) -------结合了以上的两个算法,为了避免缺陷而提出。标记阶段和Mark-Sweep算法相同,标记后不是清理对象,而是将存

活对象移向内存的另一端。然后清除边界外的对象。如图

分代收集算法(Generational Collection)重要

分代收集算法是目前大部分JVM所采用的算法,其核心思想是根据对象存活的不同生命周期将内存划分为不同的域,一般情况下将GC堆划分为老年代和新生代。

老年代的特点就是在垃圾回收时只有少量的对象需要被回收。

新生代的特点就是在回收时都有大量垃圾需要被回收,因此可以根据不同的区域选择不同的算法。

目前大部分的jvm的GC对于新生代都采用Copying算法,因为新生代中每次垃圾回收都要回收大部分对象,即要复制的操作比较少,当通常不是按照1:1划分新生代。一般将新生代划分为一块较大的Eden空间和两个较小的Suivior空间,每次使用Eden空间和其中的一块Survivor空间,当进行回收时,将该两块空间中还存活的对象复制到另一块Survivor空间中。

相关推荐
suweijie7682 小时前
SpringCloudAlibaba | Sentinel从基础到进阶
java·大数据·sentinel
公贵买其鹿3 小时前
List深拷贝后,数据还是被串改
java
xlsw_6 小时前
java全栈day20--Web后端实战(Mybatis基础2)
java·开发语言·mybatis
神仙别闹7 小时前
基于java的改良版超级玛丽小游戏
java
黄油饼卷咖喱鸡就味增汤拌孜然羊肉炒饭7 小时前
SpringBoot如何实现缓存预热?
java·spring boot·spring·缓存·程序员
暮湫8 小时前
泛型(2)
java
超爱吃士力架8 小时前
邀请逻辑
java·linux·后端
南宫生8 小时前
力扣-图论-17【算法学习day.67】
java·学习·算法·leetcode·图论
转码的小石8 小时前
12/21java基础
java
李小白668 小时前
Spring MVC(上)
java·spring·mvc