Java内存模型和内存溢出异常

 

Java内存模型和内存溢出异常

1、程序计数器:线程私有的,可以看作是当前线程所执行的字节码的行号指示器。由于java多线程是通过线程的轮流切换并分配处理器时间的方式来实现的。所以在切换后要恢复到正确的执行位置,需要程序计数器来记录。

2、java虚拟机栈:线程私有的,生命周期和线程相同。虚拟机栈描述的是java方法执行的内存模型。因为每个方法在执行的时候会创建一个栈帧,用于存储局部变量表,操作数栈,动态链接,方法出口等信息。

局部变量表中存放了编译器可知的各种基本数据类型以及对象引用和returnAddress类型(指向一条字节码指令的地址)。他所需的内存空间在编译期间完成分配。

虚拟机栈规定了两种异常状况:

如果线程请求的栈深度大于虚拟机所允许的深度,抛出*异常。

public AdjustQueryRequest setCurrentErp(String currentErp) {

    setCurrentErp(currentErp);

    return this;

}

 

如果虚拟机栈可以动态扩展,当扩展时无法申请到足够的内存,将会抛出OutOfMemoryError异常

在多线程的情况下,当创建的线程过多的时候,内存有限,容易抛出OutOfMemoryErrorgnahciy异常


3、本地方法栈:发挥的作用和虚拟机栈类似,不过他为虚拟机用到的本地方法服务。

4、java堆:线程共享,在虚拟机启动时创建,基本上所有的对象实例都在堆上创建,是gc的主要区域。从内存分配的角度来说,线程共享的java堆可能划分出多个线程私有的分配缓冲区(TLAB)。java堆可以处于物理上不联系的内存空间中。如果在堆中没有内存完成实例分配,并且堆也无法扩展的时候,将会抛出OutOfMemoryError异常

public class Test {

    static class OOMObject{}

 

    public static void main(String[] args) {

        List<OOMObject> list=new ArrayList<>();

        while (true){

            list.add(new OOMObject());

        }

    }

}

 

5、方法区:线程共享,用于存储已经被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据。

当方法区无法满足内存分配的要求的时候,抛出OutOfMemoryError异常

 

如何操作对象呢?


Java内存模型和内存溢出异常

两种对象的访问方法一种是通过句柄访问,如上图,还有一种是通过直接指针的方式访问,局部变量表中的引用直接指向java堆中的数据实例,需要在数据实例哪里封装一层,去指向对象的类型数据。