码迷,mamicode.com
首页 > 其他好文 > 详细

jvm相关知识点学习(一)

时间:2019-07-20 10:03:16      阅读:78      评论:0      收藏:0      [点我收藏+]

标签:erro   假设   区域   一个   标识   round   res   就是   回收   

1、Class loader类加载器: 

      负责加载class文件,class文件在文件开头有特定的文件标识,并且classLoader只负责class文件的加载,至于它是否可以运行,则由Execution English 决定

      技术图片

2、Native Interfice

      本地接口的作用是融合不同的编程语言为java 所用,它的初衷是融合 c/c++程序。它的具体做法是Native Method Stack中登记native方法,在Execution Engine执行时加载native libraies.

     目前该方法使用的越来越少,除非是与硬件有关的应用。

3、method Area方法区

      方法区是被所有的线程共享,所有的字段和方法字节码,以及一些特殊方法如构造函数,接口代码也在此进行定义,简单说,所有定义的方法的信息都保存在该区域,此区属于共享区间。

      静态变量+常量+类信息+运行时常量池存在方法区中+实例变量存在堆内存中

4、PC Register 程序计数器

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

5、Native Method Stack 本地方法栈

     它的具体做法是Native Method Stack中登记的native方法,在Execution Engine执行时加载native libraise.

6、Stack栈是什么

      栈也叫栈内存,主管java 程序的运行,是在线程创建时创建,它的生命期是跟随线程的生命期,线程结束栈内存也就会释放,对于栈来说不存在垃圾回收的问题,只要线程一结束该栈就over,生命周期和线程一致,是线程私有的。基本类型的变量和对象的引用变量都是在函数的栈内存中分配。

6.1、栈存储什么

      栈帧中主要保存3类数据:

      本地变量(local variables):输入参数和输出参数以及方法内的变量

     栈操作(Operand Stack): 记录出栈、入栈的操作

     栈帧数据(Frame Data): 包括类文件方法等

6.2 、栈运行的原理

    栈中的数据都是以栈帧(Stack Frame)的格式存在,栈帧是一个内存区块,是一个数据集,是一个有关方法(Method)和运行期数据的数据集,当一个方法A被调用时就产生了一个栈帧F1,并被压入到栈中,A方法有调用了B方法,于是产生的栈帧F2也被压入栈,B方法又调用了C方法,于是产生的栈帧F3也被压入栈,。。。。。执行完毕后,先弹出F3栈帧,再弹出F2栈帧,在弹出F1栈帧,遵循着“先进后出”“后进先出”的原则

三种JVM

sun公司的HotSpot

BEA公司的JRockit

IBM公司的J9 VM

堆内存示意图:

       技术图片

技术图片

  

注:伊侚区(Eden Space) 幸存0区(Survivor 0 Space) 幸存1区(Survivor 1 Space) 养老区(Tenure Generation Space) 永久存储区(Permanent Space)

7.1 新生区

     新生区是类的诞生、成长、消亡的区域,一个类在这里产生,应用,最后被垃圾回收器收集,结束生命。新生区又分为两部分:伊甸区(Eden space )和幸存者区(Survicor space),所有的类都是在伊甸区被new 出来的。幸存区有两个:0区(Survicor 0 space)和 1区(Survivor 1 space)。当伊甸园的空间用完时,程序有需要创建对象,JVM的垃圾回收器将对伊甸园区进行垃圾回收(Minor GC),将伊甸园区中的不在被其他对象引用的对象进行销毁。然后将伊甸园中剩余对象移动到幸存0区。若幸存0区也满了,在对该区进行垃圾回收,然后移动到1区。如果1区也满了,在移动到养老区。若养老区也满了,那么这个时候产生Major GC(FullGC ),进行养老区内存清理。若养老区执行了Full GC之后发现依然无法进行对象的保存,就会产生OOM异常“outofmemoyrError”

如果出现java.lang.OutofMemoryError:Java heap space异常,说明java虚拟机的堆内存不够。原因有二:

  (1)java虚拟机的堆内存设置的不够,可以通过参数-Xms、-Xmx来调整。

   (2)代码中创建了大量的大对象,并且长时间不能被垃圾收集器手机(存在被引用)

7.2 养老区

     养老区用于保存从新生区筛选出来的JAVA对象,一般池对象都在这个区域活跃。

 7.3 永久区

    永久存储区是一个常驻内存区域,用于存放JDK自身所携带的Class,Interface的元数据,也就是说它存储的是运行环境必须的类信息,被装载进此区域的数据是不会被垃圾回收器回收掉的,关闭JVM才会释放此区域所占的内存。

    如果出现java.lang.outofmemoryError:PermGen space,说明是java虚拟机对永久代Prem内存设置不够。一般出现这种情况,都是程序启动需要加载大量的第三方jar包。例如:在一个Tomcat下部署了太多的应用。或者大量动态反射生成的类不断的被加载,最终导致Prem区被占满。

JDK1.6及以前:有永久代,常量池1.6在方法区

JDK1.7:有永久代,但已经逐步“去永久代”,常量池1.7在堆

JDK1.8:无永久代,常量池1.8在元空间

实际而言,方法区(Method)和对一样,是各个线程共享的内存区域,它用于存储虚拟机加载的:类信息+普通常量+静态常量+编译器编译后的代码等,虽然JVM规范将方法区描述为堆的一个逻辑部分,但它还有一个别名叫做Non-Heap(非堆),目的就是要和对分开

     对于HotSpot虚拟机,很多开发者习惯于将方法区称之为“永久代”,但严格本质上说两者是不同的,或者说使用永久代来实现方法区而已,永久代是方法区(相当于一个接口interface)的一个实现,jdk1.7的版本中,已经将原本放在永久代的字符串常量池移走。

    常量池(Constant pool)是方法区的一部分,class文件除了有类的版本、字段、方法、接口等描述信息外,还有一项信息就是常量池,这部分内容将在类加载后进入方法区的运行时常量池中存放 。

   技术图片

   技术图片

jdk 1.8之后将最初的永久代取消了,有元空间取代

目的:将HotSpot 与 JPockit 两个虚拟机的标准合起来了

堆内存调优简介:

  -Xms :  设置初始分配大小,默认为物理内存的“1/64”

 -Xmx : 最大分配内存,默认为物理内存的“1/4”

-xx+PrintGCDetail: 输出详细的GC处理日志

一般什么时候发生GC,如何处理;

  java中的GC会有两种回收: 年轻代的Minor GC,另外一种就是老年代的FUll GC,新对象创建的时候,如果伊甸园的空间不足会触发MinorGC,如果此时老年代的内存空间不足会触发FULL GC,如果空间都不足就会抛出OutOf MemoryError。

GC回收策略的理解:

年轻代(伊甸园+两个幸存区),GC回收策略为“复制”

老年代的保存空间一般较大,GC回收策略为“整理-压缩”;

技术图片

 

JVM在进行GC时,并非每次都对上面三个内存区域一起回收,大部分时候收的都是指新生代。

因此GC按照回收的区域又分为了两种类型,一种是普通GC(minor GC),一种是全局GC(major GC or FUll GC),普通GC(minor GC):只针对新生代区域的GC

 全局GC(major GC or full GC ): 针对老年代GC,偶尔伴随对新生代GC以及对永久代的GC。

GC的三种算法:

1、年轻代中使用的是Minor GC,这种GC算法采用的是复制算法(Coping)

原理:Minor GC 会把Eden中的所有活的对象都移到Survivor区域中,如果Survivor区中放不下,那么剩下的活的对象就被移到Old generation中,也即一旦收集后,Eden就变成空的了

当对象在Eden(包括一个Survivor区域,这里假设是from区域)出生后,在经过一次Minor GC后,如果对象还存活,并且能够被另外的一块Survivor区域所容纳(上面已经假设为from区域,这里应为to区域,即to区域又足够的内存空间来存储Eden和from区域中存活的对象,)则使用复制算法将这些任然还存活的对象复制到另外一块Survivor区域(即to区域),然后清理所使用过的Eden以及Survivor区域(即from区域),并且将这些对象的年龄设置为1,以后对象在Survivor区每熬过一次Minor GC,就将对象的的年龄+1,当对象的年龄达到某个值时(默认是15岁,通过-xx:maxTenuringThreshold来设定参数),这些对象就会成为老年代

-XX:MaxTenuringThreshold --设置对象在新生代中存活的次数

年轻代中的GC,主要是复制算法(Copying)

HotSpot JVM把年轻代分为了三部分:1个Eden区和2个Survivor区(分别叫from和to),默认比例是8:1:1,一般情况下,新创建的对象都会被分配到Eden区(一些大对象特殊处理),这些对象经过第一次Minor GC后,如果任然存活,将会被移到Survivor区,对象在Survivor区中每熬过一次Minor GC,年龄就会增加1岁,当他的年龄增加到一定的程度的时候,就会被移动到年老代中。因为年轻代中的对象基本都是朝生夕死的(80%)以上,所以在年轻代的垃圾回收算法使用的是复制算法,复制算法的基本思想就是将内存分为两块,每次只用其中的一块,当这一块内存用完,就将还活着的对象复制到另一块上。复制算法不会产生内存碎片。

 

   在GC开始的时候,对象只会存在于Eden区和名为“from”的Survivor区,Survivor区“to”是空的,紧接着进行GC,Eden区所有存活的对象都会被复制到“To”,而在from区中,任存活的对象会根据他们的年龄值来决定去向,年龄达到一定的值(年龄域值,可以通过-XX:MaxTenuringThreshold来设置)的对象会被移动到老年代中,没有达到阈值的对象会被复制到“To”区域,经过这次GC后,Eden和From区已经被清空,这个时候,“from”和“To”会交换他们的角色,也就是新的“To”就是上次GC前的From,新的“from”就是上次GC前的“To”,不管怎样,不管怎样都会保证名为To的Survivor区域是空的,Minor GC会一直重复这样的过程,知道“To”区被填满之后,会将多有的对象移动代老年代中。

 

    因为Eden区对象一般存活率低,一般的,使用两块10%的内存作为空闲和活动区间,而另外80%的内存,则是用来给新建对象分配内存的。一旦发生GC,将10%的活动区与另外80%中存活的对象转移到10%的空闲区间,接下来,将之前90%的内存全部的释放,一次类推。

   复制算法的优点和缺点:

    复制算法弥补了标记/清除算法中,内存布局混乱的缺点,不过与此同时,它的缺点也是相当明显的。

   1、它浪费了一半的内存,这太要命了

   2、如果对象的存活率很高,我们可以极端一点,假设是100%的存活,那么我们需要将所有的对象都复制一遍,并将所有的引用地址重复一遍。复制这一工作所花费的时间,在对象存活率达到一定程度时,将会变得不可忽视,所以从以上描述不难看出,复制算法要想使用,最起码对象的存活时间要非常低才行,而且最重要的是我们必须要克服50%的内存浪费。

标记清除/标记整理算法:FULLGC又叫MajorGC(全局GC)

老年代一般是由标记清除或者是标记清除与标记整理的混合实现。

标记清除: 1、标记:从根集合开始扫描,对存活的对象进行标记。

                    2、清除: 扫描整个内存空间,回收未被标记的对象,使用free-list记录可以标记的区域。

标记清除的优点和缺点:

             1、首先,他的缺点是效率比较低(递归与全堆对象遍历),而且在进行GC的时候,需要停止应用程序,这会导致用户体验非常的差劲。

             2、其次,主要的缺点是这种方式清理出来的空闲内存不是连续的,这点不难理解,我们的死亡对象都是随即的出现在内存的各个角落的,现在把他们清除之后,内存的布局自然会乱七八糟。而为了应付这一点,JVM就不得不维持一个内存的空闲列表。这又是一种开销,而且在分配数组对象的时候,寻找连续的内存空间会不太好找。

 标记整理:标记整理算法唯一的缺点就是效率也不高,不仅要标记所有存活的对象,还要整理所有存活对象的引用地址。从效率上来说,标记/整理算法要低于复制算法。

内存效率:复制算法>标记清除算法>标记整理算法(此处的效率只是简单的对比时间复杂度,实际情况不一定如此)

内存整齐度:复制算法=标记整理算法>标记清除算法

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

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

 

jvm相关知识点学习(一)

标签:erro   假设   区域   一个   标识   round   res   就是   回收   

原文地址:https://www.cnblogs.com/wcgstudy/p/11216475.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!