现实企业级Java开发中,有时候我们会碰到下面这些问题:
package com.deppon.tps;
public class User {
private String name;
private String sex;
private int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public User( String name,String sex,int age){
this.name=name;
this.sex=sex;
this.age=age;
}
}
然后我们创建一个main方法就是上面的Test1类里面,
创建ArrayList,并且无限向其中添加User类对象的方法是:
package com.deppon.tps;
import java.util.ArrayList;
import java.util.List;
public class Test1 {
public static void main(String[] args) {
List persons = new ArrayList ();
while( 1>0){
persons.add( new User("liuhai","male",25));
}
}
}
当运行上述代码时候,堆溢出了,产生了heap dump文件(这里用VM参数指定了如果堆溢出则产生heap dump)
如图所示,这里很清楚的表示,main()方法中有一个集合类型,,然后集合中的每一个元素都是com.charles.research.User的对象,并且每个对象的Shallow Heap和Retained Heap大小都为24byte。因为ArrayList一直存在,所以当对象足够多时候,就把heap弄满并且溢出了。这里创建了76340681个User对象,每个对象占据了24个byte,所以一共占据了24 * 7634068=183217632字节,差不多约为 174.72M 大小的堆空间。在这里 User对象的S hallow Size 和 Retained Size都为24byte呢?
我们要看下Shallow Size和Retained Size各是什么?
Shallow Size是对象本身占据的内存的大小,不包含其引用的对象。对于常规对象(非数组)的Shallow Size由其成员变量的数量和类型来定,而数组的ShallowSize由数组类型和数组长度来决定,它为数组元素大小的总和。
Retained Size=当前对象大小+当前对象可直接或间接引用到的对象的大小总和。(间接引用的含义:A->B->C,C就是间接引用) ,并且排除被GC Roots直接或者间接引用的对象
所以,我们这里的User类,因为我们机器是32位 WIN7系统,所以对象头占据8byte,它包含String对象引用(name),占据4byte,包含String对象引用(sex),占据4byte,包含一个int类型(age),占据4byte,所以一共占据8+4+4+4=20byte, 因为要补齐位数,所以最后尺寸为24byte. 这就是这个对象的本身大小(Shallow Heap)的大小。
小知识:为了说明补齐,大家也可以做实验,如果我们User中加一个String成员,那么User类大小仍然为24byte,因为这个新String对象的引用4byte刚好去填了补齐的那个坑,如果再加一个String成员,那么User类大小就直接从24byte升到32byte了,因为又产生了一个坑需要补齐。
而我们的User类中没有引入其他的类(不包括String,因为String直接被Root GC引用),所以回收User占据的内存就是回收User自身,所以Retained Heap大小等同于Shallow Heap大小。
1.打开/tomcat_home/bin/catalina.sh文件
2.加上:JAVA_OPTS="$JAVA_OPTS -server -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=D:\heapdump"
如下图位置:
注:其中不设-XX:HeapDumpPath时,dump出的文件在/tomcat_home/bin目录下
1.打开/tomcat_home/bin/catalina.bat文件
2.加上:set JAVA_OPTS=%JAVA_OPTS% -server -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=D:\heapdump
如下图位置: