Java虚拟机——内存模型中的异常分析
1.Java虚拟机简介
JVM:Java Virtual Machine(Java虚拟机)
常见虚拟机:JVM、VMwave、Virtual Box
当前Java使用的虚拟机:HotSpot(集结了之前各种版本的优点)
每次Java虚拟机版本的更新在用户层面上是看不到的,需要通过代码来感受性能的提升;
2.JDK、JRE、JVM
JDK:Java开发工具包(JDK中包含JRE)
JDK是Sun公司提供的一套用于开发Java应用程序的开发包,提供编译运行Java程序所需的各种工具和资源,包括Java编译器,Java运行时环境以及常用Java类库
JRE:Java运行环境
JVM:Java虚拟机(对应平台的虚拟机,不跨平台)
在安装jdk时,需要设置Path变量和CLASSPATH变量:
设置Path变量的目的:为了让操作系统找到指定的命令程序(因为操作系统不支持jdk中的命令,如java、javac等命令)
设置CLASSPATH变量的目的:为了让Java执行环境找到指定的class文件,以及在程序中载入和引用其他class文件;
3.Java堆内存的OOM异常(OutOfMemoryError异常)(无限创建对象)
测试程序(⭐)
public class TestOOM {
static class OOMObject {
}
//JVM 程序启动的时候通过JVM参数指定最大最小的内容
//-Xmx20m -Xms20m -XX:+HeapDumpOnOutOfMemoryError
//该参数设置点击右上角的TestOOM的Edit configurations进去设置,将上面这段复制到VM options中就完成了设置
//该程序执行后会抛出OutOfMemoryError,并且会在这个项目下自动生成一个Dump文件,用来存放已经加载过的内容(要分析这个文件,得结合eclipse的一个工具)
public static void main(String[] args) {
List<OOMObject> list = new ArrayList<>();
while(true) {
list.add(new OOMObject());
}
}
}
输出:
java.lang.OutOfMemoryError: Java heap space
Dumping heap to java_pid12376.hprof …
Heap dump file created [131902018 bytes in 0.963 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.yy.jvm.TestOOM.main(TestOOM.java:20)Process finished with exit code 1
内存泄漏与内存溢出
-
内存泄漏:对象无法被GC(无法被回收)
-
内存溢出:内存对象确实还应该存活。此时要根据JVM堆参数与物理内存相比较检查是否还应该把JVM堆内存调大;或者检查对象的生命周期是否过长(例如本应该为局部变量但把它设置为了属性)。
4.虚拟机栈的溢出(递归调用)
测试用例(⭐)
public class TestStackOF {
private int stackDeep = 1;
public void setDeep() {
stackDeep++;
setDeep();
}
//-Xss100k(自己设置的参数),这个参数用来更改栈的深度;
//该参数设置点击右上角的TestOOM的Edit configurations进去设置,将上面这段复制到VM options中就完成了设置
public static void main(String[] args) {
TestStackOF testStackOF = new TestStackOF();
try {
testStackOF.setDeep();
}catch (Throwable e) { //将要抛出的是error,所以这里不是Exception,而是他们的父类Throwable
e.printStackTrace();
System.out.println("Call deep:" + testStackOF.stackDeep);
}
}
}
输出:
java.lang.StackOverflowError
at com.yy.jvm.TestStackOF.setDeep(TestStackOF.java:11)
at com.yy.jvm.TestStackOF.setDeep(TestStackOF.java:12)
at com.yy.jvm.TestStackOF.setDeep(TestStackOF.java:12)
…(此处省略n行)
Call deep:2303