1 对象的创建
- 虚拟机遇到new指令时会进行如下3个步骤:
- 去常量池中找到这个类的引用,查看是否已被加载。
- 确认这个类已被加载、解析和初始化过。
- 如果1和2满足,则直接引用这个地址,不满足,就要执行类加载的过程。
- 类加载检查通过后,接下来虚拟机将为新生成对象分配内存。分配方式有如下两种:
- 指针碰撞:
- 假设java堆中内存是绝对规整的。
- 所有用过的内存放到一边,没用过的放到一边。
- 中间放着一个指针作为分界点的指示器。
- 分配内存就是将指针向空极限空间挪一段与对象大小相等的距离。
- 空闲列表:
- 假设java堆中已使用内存和空闲内存是相互交错的
- 虚拟机需要维护一个列表记录哪些内存是可用的
- 在分配的时候就是从可用列表中找到一块足够大的内存划分给对象实例
- 指针碰撞与空闲列表的选择依据如下:
- 由java堆是否规整决定的
- java堆是否规整由垃圾收集器是否带有压缩整理功能决定的
- Serial、ParNew等带Compact过程的收集器采用的是指针碰撞
- CMS基于Mark-Sweep算法的收集器是采用空闲列表
对象创建过程步骤如下图所示:
2 对象的内存布局
一个Java对象在内存中包括3个部分:对象头、实例数据和对齐填充,内存布局结构如下图所示:
3 对象的生命周期
3.1 内存划分结构图
3.2 对象的分配原则
- 优先在Eden分配,当没有足够空间时就进行一次MinorGC。
- “朝生夕灭”的“短命大对象”是最可怕的事情,遇到大对象可以直接放入老年代。
- 当满足一定的年龄的对象(对象年龄计数器)就将其放到老年代中。
- 当Survivor空间中相同年龄所有对象大小的总和大于Survivor空间的一半,就将其放入到老年代中。
3.3 对象的生命周期过程
对象的生命周期过程如下图所示:
4 内存溢出模拟
4.1 堆溢出
- Java堆溢出异常代码
import java.util.ArrayList;
import java.util.List;
/**
* VM Args:-Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError
*
* @author zzm
*/
public class HeapOOM {
static class OOMObject {
}
public static void main(String[] args) {
List<OOMObject> list = new ArrayList<OOMObject>();
while (true) {
list.add(new OOMObject());
}
}
}
输出结果
java.lang.OutOfMemoryError: Java heap space
Dumping heap to java_pid25504.hprof ...
Heap dump file created [28284365 bytes in 0.080 secs]
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at java.util.Arrays.copyOf(Arrays.java:3210)
at java.util.Arrays.copyOf(Arrays.java:3181)
at java.util.ArrayList.grow(ArrayList.java:261)
at java.util.ArrayList.ensureExplicitCapacity(ArrayList.java:235)
at java.util.ArrayList.ensureCapacityInternal(ArrayList.java:227)
at java.util.ArrayList.add(ArrayList.java:458)
at com.ddpyjqtd.HeapOOM.main(HeapOOM.java:20)
Process finished with exit code 1
4.2 虚拟机栈和本地方法栈溢出
- 虚拟机和本地方法栈OOM异常代码
/**
* VM Args:-Xss128k
* @author zzm
*/
public class JavaVMStackSOF {
private int stackLength = 1;
public void stackLeak() {
stackLength++;
stackLeak();
}
public static void main(String[] args) throws Throwable {
JavaVMStackSOF oom = new JavaVMStackSOF();
try {
oom.stackLeak();
} catch (Throwable e) {
System.out.println("stack length:" + oom.stackLength);
throw e;
}
}
}
不配置VM参数时输出结果
stack length:29328
Exception in thread "main" java.lang.StackOverflowError
at com.ddpyjqtd.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:13)
at com.ddpyjqtd.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:13)
at com.ddpyjqtd.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:13)
配置 -Xss128k 后输出,明显栈的长度变小了
Exception in thread "main" stack length:985
java.lang.StackOverflowError
at com.ddpyjqtd.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:12)
at com.ddpyjqtd.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:13)
- 创建线程导致内存溢出异常代码
/**
* VM Args:-Xss2M (这时候不妨设大些)
* @author zzm
*/
public class JavaVMStackOOM {
private void dontStop() {
while (true) {
}
}
public void stackLeakByThread() {
while (true) {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
dontStop();
}
});
thread.start();
}
}
public static void main(String[] args) throws Throwable {
JavaVMStackOOM oom = new JavaVMStackOOM();
oom.stackLeakByThread();
}
}
输出结果
按理应该输出
Exception in thread "main" java.lang.OOutOfMemoryError: unable to careate new native thread.
但我本机执行程序后CPU马上跑到100%,Idea卡死一直没有反应,无法验证该结果。
4.3 方法区和运行时常量池溢出
- 运行时常量池导致的内存溢出异常代码
import java.util.ArrayList;
import java.util.List;
/**
* VM Args:jdk1.6使用参数 -XX:PermSize=10M -XX:MaxPermSize=10M
jdk1.7+ 使用参数 -XX:MetaspaceSize=10M -XX:MaxMetaspaceSize=10M
*
* @author zzm
*/
public class RuntimeConstantPoolOOM {
public static void main(String[] args) {
// 使用List保持着常量池引用,避免Full GC回收常量池行为
List<String> list = new ArrayList<String>();
// 10MB的PermSize在integer范围内足够产生OOM了
int i = 0;
while (true) {
list.add(String.valueOf(i++).intern());
}
}
}
输出结果
Error occurred during initialization of VM
OutOfMemoryError: Metaspace
Process finished with exit code 1
- String.intern()返回引用的测试代码
public class RuntimeConstantPoolOOM1 {
public static void main(String[] args) {
String str1 = new StringBuilder("中国").append("钓鱼岛").toString();
System.out.println(str1.intern() == str1);
String str2 = new StringBuilder("ja").append("va").toString();
System.out.println(str2.intern() == str2);
}
}
输出结果
jdk1.6中返回
false
false
jdk1.7+中返回
true
false
- 借助CGLIB全方法区出现内存溢出异常代码
/**
* VM Args:jdk1.6使用参数 -XX:PermSize=10M -XX:MaxPermSize=10M
jdk1.7+ 使用参数 -XX:MetaspaceSize=10M -XX:MaxMetaspaceSize=10M
* @author zzm
*/
public class JavaMethodAreaOOM {
public static void main(String[] args) {
while (true) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(OOMObject.class);
enhancer.setUseCache(false);
enhancer.setCallback(new MethodInterceptor() {
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
return proxy.invokeSuper(obj, args);
}
});
enhancer.create();
}
}
static class OOMObject {
}
}
输出结果
Exception in thread "main" net.sf.cglib.core.CodeGenerationException: java.lang.reflect.InvocationTargetException-->null
at net.sf.cglib.core.AbstractClassGenerator.generate(AbstractClassGenerator.java:348)
at net.sf.cglib.proxy.Enhancer.generate(Enhancer.java:492)
at net.sf.cglib.core.AbstractClassGenerator$ClassLoaderData.get(AbstractClassGenerator.java:117)
at net.sf.cglib.core.AbstractClassGenerator.create(AbstractClassGenerator.java:294)
at net.sf.cglib.proxy.Enhancer.createHelper(Enhancer.java:480)
at net.sf.cglib.proxy.Enhancer.create(Enhancer.java:305)
at com.ddpyjqtd.JavaMethodAreaOOM.main(JavaMethodAreaOOM.java:25)
Caused by: java.lang.reflect.InvocationTargetException
at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at net.sf.cglib.core.ReflectUtils.defineClass(ReflectUtils.java:459)
at net.sf.cglib.core.AbstractClassGenerator.generate(AbstractClassGenerator.java:339)
... 6 more
Caused by: java.lang.OutOfMemoryError: Metaspace
at java.lang.ClassLoader.defineClass1(Native Method)
at java.lang.ClassLoader.defineClass(ClassLoader.java:763)
... 11 more
Process finished with exit code 1
4.本机直接内存溢出
- 使用unsafe分配本机直接内存代码
/**
* VM Args:-Xmx20M -XX:MaxDirectMemorySize=10M
* @author zzm
*/
public class DirectMemoryOOM {
private static final int _1MB = 1024 * 1024;
public static void main(String[] args) throws Exception {
Field unsafeField = Unsafe.class.getDeclaredFields()[0];
unsafeField.setAccessible(true);
Unsafe unsafe = (Unsafe) unsafeField.get(null);
while (true) {
unsafe.allocateMemory(_1MB);
}
}
}
输出结果
Exception in thread "main" java.lang.OutOfMemoryError
at sun.misc.Unsafe.allocateMemory(Native Method)
at com.ddpyjqtd.DirectMemoryOOM.main(DirectMemoryOOM.java:21)
5 引用
- 《深入理解Java虚拟机 JVM高级特性与最佳实践》-周志明著