Java内存管理简介

文章一、
-X和-XX都是非标准选项
-server:虚拟机会针对server做特定优化,包括增加内存,打开本地垃圾收集器
其他参见:http://happay99.blog.hexun.com/28574460_d.html#,

|-----|-----|---------------|---|---|
  perm    old    eden           from/survivor to
            |-----------------------|
                        yong
      |-----------------------------|
                    Heap


注:Mem={Perm+ Heap={old+ yong={eden+from+to}}}
to=from/survivor
survivorRatio= eden/survivor (一般一个整数)
survivor = yong/(sr+1+1)

Perm用来存放 Java 类及jvm的静态区域

内存溢出指old段和perm段都回收后,仍无法分配新对象内存
内存分配尝试过程如下所示:
1.若eden内存够,则申请eden
2.否则,释放eden中不活跃对象,若仍然不够把部分活跃对象放入survivor,若old区够大,则survivor区的会放到old区,否则保存survivor
3.如果old区不够,则进行完全垃圾回收
4.如果仍然不够,则抛出异常

java参数设置:
ms/mx:定义old+yong 的总尺寸
NewSize/MaxNewSize:定义yong段尺寸
PermSize/MaxPermSize:定义Perm段的尺寸
SurvivorRatio:设置SurvivorRatio
例如:
MEM_ARGS=
"-Xms512m -Xmx512m
-XX:NewSize=256m -XX:MaxNewSize=256m
-XX:PermSize=128m -XX:MaxPermSize=128m
-XX:SurvivorRatio=6"
结果如下:
YOUNG+OLD: 512M
YOUNG: 256M
Perm: 128M

OLD=512M
Eden: YOUNG*6/(6+1+1)=192M
Survivor: YOUNG/(6+1+1)=32M
Mem=YONG+OLD + Perm=640M

上文转自:http://hi.baidu.com/canghaiyisu123/blog/item/fa2b2f3fbe3ca23670cf6c1a.html

文章二、
Java内存管理简介_第1张图片

如上图所示,为Java堆中的各代分布。

1. Young(年轻代)
年轻代分三个区。一个Eden区,两个Survivor区。大部分对象在Eden区中生成。 当Eden区满时,还存活的对象将被复制到Survivor区(两个中的一个),当这个Survivor区满时,此区的存活对象将被复制到另外一个 Survivor区,当这个Survivor去也满了的时候,从第一个Survivor区复制过来的并且此时还存活的对象,将被复制年老区 (Tenured。需要注意,Survivor的两个区是对称的,没先后关系,所以同一个区中可能同时存在从Eden复制过来 对象,和从前一个Survivor复制过来的对象,而复制到年老区的只有从第一个Survivor去过来的对象。而且,Survivor区总有一个是空 的。
2. Tenured(年老代,上文提到的Old)
年老代存放从年轻代存活的对象。一般来说年老代存放的都是生命期较长的对象。
3. Perm(持久代)
用于存放静态文件,如今Java类、方法等。持久代对垃圾回收没有显著影响,但是有些应用可 能动态生成或者调用一些class,例如Hibernate等,在这种时候需要设置一个比较大的持久代空间来存放这些运行过程中新增的类。持久代大小通过 -XX:MaxPermSize=进行设置。

举个例子:当在程序中生成对象时,正常对象会在年轻代中分配空间,如果是过大的对象也可能会 直接在年老代生成(据观测在运行某程序时候每次会生成一个十兆的空间用收发消息,这部分内存就会直接在年老代分配)。年轻代在空间被分配完的时候就会发起 内存回收,大部分内存会被回收,一部分幸存的内存会被拷贝至Survivor的from区,经过多次回收以后如果from区内存也分配完毕,就会也发生内 存回收然后将剩余的对象拷贝至to区。等到to区也满的时候,就会再次发生内存回收然后把幸存的对象拷贝至年老区。
通常我们说的JVM内存回收总是在指堆内存回收,确实只有堆中的内容是动态申请分配的,所以以上对象的年轻代和年老代都是指的JVM的Heap空间,而持久代则是之前提到的Method Area,不属于Heap。
了解完这些之后,以下的转载一热衷于钻研技术的哥们Richen Wang关于内存管理的一些建议——
1、手动将生成的无用对象,中间对象置为null,加快内存回收。
2、对象池技术 如果生成的对象是可重用的对象,只是其中的属性不同时,可以考虑采用对象池来较少对象的生成。如果有空闲的对象就从对象池中取出使用,没有再生成新的对象,大大提高了对象的复用率。
3、JVM调优 通过配置JVM的参数来提高垃圾回收的速度,如果在没有出现内存泄露且上面两种办法都不能保证内存的回收时,可以考虑采用JVM调优的方式来解决,不过一定要经过实体机的长期测试,因为不同的参数可能引起不同的效果。如-Xnoclassgc参数等。
推荐的两款内存检测工具
1、jconsole JDK自带的内存监测工具,路径jdk bin目录下jconsole.exe,双击可运行。连接方式有两种,第一种是本地方式如调试时运行的进程可以直接连,第二种是远程方式,可以连接以服务 形式启动的进程。远程连接方式是:在目标进程的jvm启动参数中添加-Dcom.sun.management.jmxremote.port=1090 -Dcom.sun.management.jmxremote.ssl=false -Dcom.sun.management.jmxremote.authenticate=false 1090是监听的端口号具体使用时要进行修改,然后使用IP加端口号连接即可。通过该工具可以监测到当时内存的大小,CPU的使用量以及类的加载,还提供 了手动gc的功能。优点是效率高,速度快,在不影响进行运行的情况下监测产品的运行。缺点是无法看到类或者对象之类的具体信息。使用方式很简单点击几下就 可以知道功能如何了,确实有不明白之处可以上网查询文档。
2、JProfiler 收费的工具,但是到处都有破解办法。安装好以后按照配置调试的方式配置好一个本地的session即可运行。可以监测当时的内存、CPU、线程等,能具体 的列出内存的占用情况,还可以就某个类进行分析。优点很多,缺点太影响速度,而且有的类可能无法被织入方法,例如我使用jprofiler时一直没有备份 成功过,总会有一些类的错误。

你可能感兴趣的:(java)