标签:

java内存区域

javacpp 内存参数 对于java内存配置参数_javacpp 内存参数

一些基本概念

1.方法区和堆是所有线程共享的内存区域;而java栈、本地方法栈和程序员计数器是运行时线程私有的内存区域。

2.Java堆(Heap),是Java虚拟机所管理的内存中最大的一块。Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。

3.方法区(Method Area),方法区(Method Area)与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。

4.程序计数器(Program Counter Register),程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码的行号指示器。

5.JVM栈(JVM Stacks),与程序计数器一样,Java虚拟机栈(Java Virtual Machine Stacks)也是线程私有的,它的生命周期与线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法被执行的时候都会同时创建一个栈帧(Stack Frame)用于存储局部变量表、操作栈、动态链接、方法出口等信息。每一个方法被调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。

6.本地方法栈(Native Method Stacks),本地方法栈(Native Method Stacks)与虚拟机栈所发挥的作用是非常相似的,其区别不过是虚拟机栈为虚拟机执行Java方法(也就是字节码)服务,而本地方法栈则是为虚拟机使用到的Native方法服务。

堆中存的是对象。栈中存的是基本数据类型和堆中对象的引用

javacpp 内存参数 对于java内存配置参数_java 新区 老区_02

类的生命周期

1.加载,查找并加载类的二进制数据,在Java堆中也创建一个java.lang.Class类的对象

2.连接,连接又包含三块内容:验证、准备、解析。1)验证,文件格式、元数据、字节码、符号引用验证;2)准备,为类的静态变量分配内存,并将其初始化为默认值;3)解析,把类中的符号引用转换为直接引用

3.初始化,为类的静态变量赋予正确的初始值

4.使用,new出对象程序中使用

5.卸载,执行垃圾回收

引用类型

对象引用类型分为强引用、软引用、弱引用和虚引用

强引用:就是我们一般声明对象是时虚拟机生成的引用,强引用环境下,垃圾回收时需要严格判断当前对象是否被强引用,如果被强引用,则不会被垃圾回收

软引用:软引用一般被做为缓存来使用。与强引用的区别是,软引用在垃圾回收时,虚拟机会根据当前系统的剩余内存来决定是否对软引用进行回收。如果剩余内存比较紧张,则虚拟机会回收软引用所引用的空间;如果剩余内存相对富裕,则不会进行回收。换句话说,虚拟机在发生OutOfMemory时,肯定是没有软引用存在的。

弱引用:弱引用与软引用类似,都是作为缓存来使用。但与软引用不同,弱引用在进行垃圾回收时,是一定会被回收掉的,因此其生命周期只存在于一个垃圾回收周期内。

强引用不用说,我们系统一般在使用时都是用的强引用。而“软引用”和“弱引用”比较少见。他们一般被作为缓存使用,而且一般是在内存大小比较受限的情况下做为缓存。因为如果内存足够大的话,可以直接使用强引用作为缓存即可,同时可控性更高。因而,他们常见的是被使用在桌面应用系统的缓存。

JVM参数配置

JVM启动模式

Client模式:启动速度较快,但运行时性能和内存管理效率不高

Server模式:启动比Client模式慢10%,但运行时性能和内存管理效率较高

在JVM启动的时候指定:-Client或者-Server来判断启动模式

分代收集算法 把对象分为新生代、老年代、持久代,对不同生命周期的对象使用不同的算法 http://www.importnew.com/19255.html

试想,在不进行对象存活时间区分的情况下,每次垃圾回收都是对整个堆空间进行回收,花费时间相对会长,同时,因为每次回收都需要遍历所有存活对象,但实际上,对于生命周期长的对象而言,这种遍历是没有效果的,因为可能进行了很多次遍历,但是他们依旧存在。因此,分代垃圾回收采用分治的思想,进行代的划分,把不同生命周期的对象放在不同代上,不同代上采用最适合它的垃圾回收方式进行回收。

javacpp 内存参数 对于java内存配置参数_老年代_03

新生代

所有新生成的对象首先都是放在年轻代的。年轻代的目标就是尽可能快速的收集掉那些生命周期短的对象。年轻代分三个区。一个Eden区,两个Survivor区(默认,可改),大部分对象在Eden区中生成。当Eden区满时,还存活的对象将被复制到Survivor区(两个中的一个),当这个Survivor区满时,此区的存活对象将被复制到另外一个Survivor区,当这个Survivor去也满了的时候,从第一个Survivor区复制过来的并且此时还存活的对象,将被复制“年老区(Tenured)”。需要注意,Survivor的两个区是对称的,没先后关系,所以同一个区中可能同时存在从Eden复制过来 对象,和从前一个Survivor复制过来的对象,而复制到年老区的只有从第一个Survivor去过来的对象。而且,Survivor区总有一个是空的。同时,根据程序需要,Survivor区是可以配置为多个的(多于两个),这样可以增加对象在年轻代中的存在时间,减少被放到年老代的可能

老年代

在年轻代中经历了N次垃圾回收后仍然存活的对象,就会被放到年老代中。因此,可以认为年老代中存放的都是一些生命周期较长的对象。

持久代

用于存放静态文件,如今Java类、方法等。持久代对垃圾回收没有显著影响

垃圾回收算法 http://www.importnew.com/18740.html

引用计数(Reference Counting):

比较古老的回收算法。原理是此对象有一个引用,即增加一个计数,删除一个引用则减少一个计数。垃圾回收时,只用收集计数为0的对象。此算法最致命的是无法处理循环引用的问题。

标记-清除(Mark-Sweep):

此算法执行分两阶段。第一阶段从引用根节点开始标记所有被引用的对象,第二阶段遍历整个堆,把未标记的对象清除。此算法需要暂停整个应用,同时,会产生内存碎片。 碎片太多可能引发另一次GC

javacpp 内存参数 对于java内存配置参数_javacpp 内存参数_04

复制(Copying):

此算法把内存空间划为两个相等的区域,每次只使用其中一个区域。垃圾回收时,遍历当前使用区域,把正在使用中的对象复制到另外一个区域中。次此法每次只处理正在使用中的对象,因此复制成本比较小,同时复制过去以后还能进行相应的内存整理,不会出现“碎片”问题。当然,此算法的缺点也是很明显的,就是需要两倍内存空间。

javacpp 内存参数 对于java内存配置参数_java 新区 老区_05

标记-整理(Mark-Compact):

此算法结合了“标记-清除”和“复制”两个算法的优点。也是分两阶段,第一阶段从根节点开始标记所有被引用对象,第二阶段遍历整个堆,把清除未标记对象并且把存活对象“压缩”到堆的其中一块,按顺序排放。此算法避免了“标记-清除”的碎片问题,同时也避免了“复制”算法的空间问题。

javacpp 内存参数 对于java内存配置参数_垃圾回收_06

串行GC

Serial收集器

1)特点:

–仅仅使用单线程进行内存回收;

–它是独占式的内存回收 ;

–进行内存回收时, 暂停所有的工作线程(“Stop-The-World”) ;

–使用复制算法;

–适合CPU等硬件一般的场合;

–到JDK1.7为止,是JVM Client模式下默认的新生代收集器;

2)设置参数:

-XX:+UseSerialGC 指定使用新生代Serial和老年代SerialOld

SerialOld收集器

javacpp 内存参数 对于java内存配置参数_老年代_07

1)特点:

–同新生代Serial收集器一样,单线程、独占式的垃圾收集器;

–使用“标记-整理”算法;

–通常老年代内存回收比新生代内存回收要更长时间,所以可能会使应用程序停顿较长时间;

2)设置参数:

-XX:+UseSerialGC 新生代、老年代都使用串行GC;

-XX:+UseParNewGC 新生代使用ParNew,老年代使用SerialOld;

-XX:+UseParallelGC 新生代使用Parallel,老年代使用SerialOld;

并行GC(吞吐量优先)

ParNew收集器

1)特点:

–Serial的多线程版本;

–使用复制算法 ;

–垃圾回收时,应用程序仍会暂停,只不过由于是多线程回收,在多核CPU上,回收效率会高于串行GC。反之在单核CPU,效率会不如串行GC;

2)设置参数:

-XX:+UseParNewGC 新生代使用ParNew,老年代使用SerialOld;

-XX:+UseConcMarkSweepGC 新生代使用ParNew,老年代使用CMS;

-XX:ParallelGCThreads=n 指定ParNew收集器工作时的收集线程数,当CPU核数小于8时,默认开启的线程数等于CPU数量,当高于8时,可使用公式:3+((5*CPU_count)/8) 。

ParNew收集器其实就是Serial收集器的多线程版本,除了多条线程收集之外,其余行为包括Serial收集器可用的设置参数、收集算法、Safepoint、对象分配规则、回收策略等都与Serial收集器完全一样,并没有太多特别之处。但它却是运行在JVM Service模式下首选的新生代收集器,其中一个很重要的原因是:除了Serial收集器外,目前只有它能于CMS收集器(并发GC)配合工作。

ParNew收集器在单CPU环境中绝对不会有比Serial收集器更好的效果,甚至由于存在线程切换的开销,ParNew收集器在两个CPU环境中都不能100%保证优于Serial收集器。随着可以使用的CPU数量的增加,它对于GC时系统资源的有效利用还是有利的。

Parallel收集器

1)特点:

–同ParNew回收器一样, 不同的地方在于,它非常关注系统的吞吐量(通过参数控制) ;

–使用复制算法;

–支持自适应的GC调节策略;

2)设置参数:

-XX:+UseParallelGC 新生代使用Parallel,老年代使用SerialOld;

-XX:+UseParallelOldGC 新生代使用Parallel,老年代使用ParallelOld;

-XX:MaxGCPauseMillis=n 设置内存回收的最大停顿时间,单位ms ;

-XX:GCTimeRatio=n 设置吞吐量的大小,假设值为n(在0-100之间),那么系统将花费不超过1/(n+1)的时间用于内存回收。默认值为99,就是允许最大1%的垃圾收集时间;

-XX:+UseAdaptiveSizePolicy 自适应GC策略的开关参数。

吞吐量(Throughput) = 运行用户代码时间 /(运行用户代码时间 + 垃圾收集时间)。

Parallel收集器支持“GC自适应的调节策略(GC-Ergonomics)”,这也是Parallel收集器与ParNew收集器的一个重要区别。

设置参数-XX:+UseAdaptiveSizePolicy,这是一个开关参数,当这个参数打开后,不需要手工指定新生代的大小(-Xmn)、Eden与Survivor区比例(-XX:SurvivorRatio)、晋升老年代对象年龄(-XX:PretenureSizeThreshold)等细节参数,虚拟机会根据当前系统的运行情况收集性能监控信息,动态调整这些参数以提供最适合的停顿时间或者最大的吞吐量,这种调节方式称为“GC自适应的调节策略”。

你也可以在使用Parallel收集器自适应调节策略时,把基本的内存数据设置好(如-Xmx 最大堆),然后使用MaxGCPauseMillis参数(更关注最大停顿时间)或GCTimeRatio(更关注吞吐量)参数给虚拟机设立一个优化目标,具体细节参数的调节任务就交由虚拟机去完成。

ParallelOld收集器

javacpp 内存参数 对于java内存配置参数_垃圾回收_08

1)特点:

–关注吞吐量的老年代并发收集器;

–使用“标记-整理”算法;

2)设置参数:

-XX:+UseParallelOldGC 新生代使用Parallel,老年代使用ParallelOld。

这个收集器是在JDK1.6中才开始提供,在此之前,如果新生代选择了Parallel收集器,老年代除了SerialOld收集器外别无选择。在注重吞吐量以及CPU资源敏感的场合,都可以优先考虑Parallel加ParallelOld收集器。

并发GC

CMS收集器

javacpp 内存参数 对于java内存配置参数_垃圾回收_09

1)特点:

–非独占式的老年代并发收集器,大部分时候应用程序不会停止运行;

–使用“标记-清除”算法,因此回收后会有内存碎片,可设置参数进行内存碎片的压缩整理 ;

–与Parallel和ParallelOld不同,CMS主要关注系统停顿时间;

2) 缺点:

对CPU资源敏感;

无法处理浮动垃圾(Floating Garbage);

内存碎片问题

其中初始标记、重新标记这两个步骤仍需“Stop-The-World”,这两个阶段是独占的,不能与用户线程一起执行,而其它阶段则可以与用户线程一起执行。

初始标记仅仅只是标记一下GC Roots能直接关联到的对象,速度很快;

并发标记阶段就是进行GC Roots Tracing的过程;

重新标记阶段则是为了修正并发标记期间因用户程序继续运作而导致标记产生变动的那一部分对象的标记记录,这个阶段的停顿时间一般会比初始标记阶段稍长一些,但远比并发标记的时间短。

由于整个过程中耗时最长的并发标记和并发清除过程,收集线程都可以与用户线程一起工作,所以,从总体上来说,CMS收集器的内存回收过程是与用户线程一起并发执行的。

浮动垃圾: 并发清理阶段用户线程还在运行着,伴随程序运行自然就还会有新的垃圾不断产生

2)设置参数:

-XX:-CMSPrecleaningEnabled  关闭预清理,默认在并发标记后会有一个预清理的操作;

-XX:+UseConcMarkSweepGC   老年代使用CMS,新生代使用ParNew;

-XX:ConcGCThreads=n   设置并发线程数;

-XX:ParallelCMSThreads=n   同上,设置并发线程数;

-XX:CMSInitiatingOccupancyFraction=n 指定老年代回收阀值,默认值为68;

-XX:+UseCMSCompactAtFullCollection 开启内存碎片整理;

-XX:CMSFullGCsBeforeCompation=n  指定进行多少次CMS垃圾回收后再进行一次内存压缩;

-XX:+CMSParallelRemarkEnabled   在使用UseParNewGC参数的情况下,尽量减少 mark(标记)的时间;

-XX:+UseCMSInitiatingOccupancyOnly  表示只有达到阀值时才进行CMS垃圾回收

javacpp 内存参数 对于java内存配置参数_软引用_10

对象何时进入老年代

长期存活的对象将进入老年代

-XX:MaxTenuringThreshold=n  假设值为n,则新生代的对象最多经历n次GC,就能晋升老年代,但这个必不是晋升的必要条件

-XX:TargetSurvivorRatio=n   用于设置Survivor区的目标使用率,即当Survivor区GC后使用率超过这个值,就可能会使用较小的年龄作为晋升年龄。默认为50

虚拟机采用分代收集的思想管理内存,那内存回收时就必须能识别那些对象该放到新生代,那些该到老年代中。为了做到这点,虚拟机为每个对象定义了一个对象年龄(Age,由GC的次数决定)。每经过一次新生代GC后仍然存活,将对象的Age增加1岁,当年龄到一定程度(默认为15)时,将会被晋升到老年代中。

如果将 -XX:MaxTenuringThreshold 参数设置为0的话,则新生代对象不经过Survivor区,直接进入老年代,对于需要大量常驻内存的应用,这样做可以提高效率;如果将此值设置为一个较大值,则新生代对象会在Survivor区进行多次复制,这样做可以增加对象在新生代的存活时间,增加对象在新生代被垃圾回收的概率,减少Full GC的频率,可以在某种程度上提高服务稳定性。

适龄对象也可能进入老年代

为了能更好地适应不同程序的内存状况,虚拟机并不是永远地要求对象年龄必须达到 MaxTenuringThreshold 才能晋升老年代,如果在 Survivor 空间中相同年龄所有对象大小的总和大于 Survivor 空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代,无须等到 MaxTenuringThreshold 中要求的年龄。

大对象直接进入老年代

-XX:PretenureSizeThreshold 即对象的大小大于此值,就会绕过新生代,直接在老年代分配(即所谓“大对象直接进入老年代”)。

除年龄外,对象体积也会影响对象的晋升的。若对象体积太大,新生代无法容纳这个对象,则这个对象可能就会直接晋升至老年代。特别是,如果程序中经常出现“短命的大对象”,容易发生内存还有不少空间却不得不提前触发Full GC以获取足够的连续空间,导致GC效率低下。可通过以下参数使用对象直接晋升至老年代的阈值,单位是byte

PretenureSizeThreshold 参数的意义在于,若遇上述情况时,能避免在 Eden 及两个 Survivor 之间发生大量的内存复制。此参数只对串行GC以及ParNew有效,而Parallel并不认识这个参数。Parallel收集器一般并不需要特别设置。如果遇到必须使用此参数的场合,也可以考虑 ParNew加CMS的收集器组合。