java jvm是什么_java:JVM及相关概念

public class TestDemo{

public static void main(String args[]){

String str = null;

str.length();

}

}

上面程序会报空指针异常,如下图,在报的这个异常中,有一行日志 at TestDemo,main(TestDemo.java:4) 代表程序运行到TestDemo 中main()函数第四行的时候发生的错误,就是通过程序计数器来记录这个程序运行的位置的。

e8fb684889c03445cd8b8ad3dc500df9.png

3.5 本地方法栈

和虚拟机栈类似,不过本地方法栈里面运行的方法不是用java写的,一般是用c或c++写的为虚拟机使用到Native方法服务,也有类似栈帧的的概念。

da08a583be4d5a4814090fff18dd961c.png

用一张图描绘每个区域存储的内容

4、内存模型和垃圾回收

正式阅读之前需要了解相关概念:Java 堆内存分为新生代和老年代,新生代中又分为1个 Eden 区域 和 2个 Survivor 区域。

4.1 概述

jvm 中,程序计数器、虚拟机栈、本地方法栈都是随线程而生随线程而灭,栈帧随着方法的进入和退出做入栈和出栈操作,实现了自动的内存清理,因此,我们的内存垃圾回收主要集中于 java 堆和方法区中,在程序运行期间,这部分内存的分配和使用都是动态的。

acf53f0b36d84d34dd383f3a50239b08.png

4.2 GC的对象

需要进行回收的对象就是已经没有存活的对象,判断一个对象是否存活有两种方法:引用计数和可达性分析

引用计数:每一个对象都有一个引用计数属性,新增一个引用时计数增加1,释放引用时计数减1。计数为0时可以回收。此方法简单,但无法解决对象互相引用的问题。

2dcfff7500c1e1861fd6d027943a42ce.png

从上图中可以看出,如果不小心把Obj1-reference 和 Obj2-reference置null。则在java堆中的两块内存依然保持互相引用无法回收。

可达性分析:从GC Root开始向下搜索,搜索所走过的路径称之为引用链。当一个对象到GC Root没有任何引用链相连时,则证明此对象不可用,不可达对象。

dcb4fa253c90348c049cdcf27262d8c9.png

可作为 GC Roots 的对象:

虚拟机栈(栈帧中的本地变量表)中引用的对象

方法区中类静态属性引用的对象

方法区中常量引用的对象

本地方法栈中 JNI(即一般说的 Native 方法) 引用的对象

4.3 什么时候触发GC

1.程序调用System.gc时可触发

2.系统自身决定触发GC的时机(根据Eden和From Space区的内存大小来决定。当内存大小不足时,则会启动GC线程并停止应用线程)

GC又分为Minor gc和Full gc

Minor gc:当Eden区满是触发,Minor gc

Full gc:

a.调用System.gc时,系统建议执行Full GC,但是不必然执行

b.老年代空间不足

c.方法区空间不足

d.通过Minor GC后进入老年代的平均大小大于老年代的可用内存

e.由Eden区、From Space区向To Space区复制时,对象大小大于To Space可用内存,则把该对象转存到老年代,且老年代的可用内存小于该对象大小

GC主要做了清理对象,整理内存的工作。Java堆分为新生代和老年代,采用了不同的回收方式。(回收方式即回收算法详见后文)

4.3.1 回收方法区:

在堆中,尤其是在年轻代中,一次垃圾回收一般可以回收 70% ~ 95% 的空间,而永久代的垃圾收集效率远低于此。永久代垃圾回收主要两部分内容:废弃的常量和无用的类。

判断废弃常量:一般是判断没有该常量的引用。

判断无用的类:要以下三个条件都满足

该类所有的实例都已经回收,也就是 Java 堆中不存在该类的任何实例

加载该类的 ClassLoader 已经被回收

该类对应的 java.lang.Class 对象没有任何地方呗引用,无法在任何地方通过反射访问该类的方法

4.4 GC算法

GC常用算法有:标记-清除算法,标记-压缩算法,复制算法,分代收集算法。

目前主流的JVM(HotSpot)采用的是分代收集算法。

分代收集算法

现在的虚拟机垃圾收集大多采用这种方式,它根据对象的生存周期,将堆分为新生代(Young)和老年代(Tenure)。在新生代中,由于对象生存期短,每次回收都会有大量对象死去,那么这时就采用复制算法。老年代里的对象存活率较高,没有额外的空间进行分配担保,所以可以使用标记-整理 或者 标记-清除。

具体过程:新生代(Young)分为Eden区,From区与To区

65391eb902e5aeb4e6672b021778b799.png

当系统创建一个对象的时候,总是在Eden区操作,当这个区满了,那么就会触发一次YoungGC,也就是年轻代的垃圾回收。一般来说这时候不是所有的对象都没用了,所以就会把还能用的对象复制到From区。

ca2ef1b338b69032607cd56d77718d4e.png

这样整个Eden区就被清理干净了,可以继续创建新的对象,当Eden区再次被用完,就再触发一次YoungGC,然后呢,注意,这个时候跟刚才稍稍有点区别。这次触发YoungGC后,会将Eden区与From区还在被使用的对象复制到To区,

6e36b3f523100bbabcaf89ab34cb15d3.png

再下一次YoungGC的时候,则是将Eden区与To区中的还在被使用的对象复制到From区。

944d2ec76638c7284ad16e9a3299b394.png

经过若干次YoungGC后,有些对象在From与To之间来回游荡,这时候From区与To区亮出了底线(阈值),这些家伙要是到现在还没挂掉,对不起,一起滚到(复制)老年代吧。

327c19404d0f1137b51351070db0c3f5.png

老年代经过这么几次折腾,也就扛不住了(空间被用完),好,那就来次集体大扫除(Full GC),也就是全量回收。如果Full GC使用太频繁的话,无疑会对系统性能产生很大的影响。所以要合理设置年轻代与老年代的大小,尽量减少Full GC的操作。

参考:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值