JVM的内存模型

本文详细介绍了JVM的内存模型,包括线程私有的程序计数器、虚拟机栈和本地方法栈,以及线程共享的Java堆和方法区。每个区域的功能、异常情况和对象实例分配等进行了阐述,还提到了直接内存和运行时常量池的相关内容。
摘要由CSDN通过智能技术生成

上一节我们讲了类的加载到卸载的使用过程以及双亲委派的详情,接下来我们继续了解JVM的内存模型。

JVM 内存区域主要分为线程私有区域:程序计数器、虚拟机栈、本地方法区】、线程共享区域:【JAVA 堆、方法区】、直接内存。

1、程序计数器(线程私有区域

程序计数器(Program Counter Register)是一块较小的内存空间,它可以看做是当前线程所执行的字节码的行号指示器。在Java虚拟机的概念模型中,字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,它是程序控制流的指示器、分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。

由于Java虚拟机的多线程是通过线程轮流切换、分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器(对于多核处理器来说是一个内核)都只会执行一条线程中的指令。因此,为了线程切换后能恢复到正确的执行位置,每条线程都需要一个独立的程序计数器,各条线程之间计数器互不影响,独立存储,我们称这类内存区域为:“线程私有”的内存。

如果线程正在执行的是一个Java方法,这个程序计数器记录的是正在执行的虚拟机字节码指令的地址。如果正在执行的是本地(Native)方法,这个程序计数器则因为空(Undefined)。此内存区域唯一一个在《Java虚拟机规范》中没有规定任何OutOfMemoryError情况的区域

2、Java虚拟机栈(线程私有区域)

与程序计数器一样,Java虚拟机栈(Java Virtual Machine Stack)也是线程私有的,它的生命周期与线程相同。虚拟机栈描述的是Java方法执行的线程内存模型:每个方法被执行的时候,Java虚拟机都会同步创建一个栈帧用于存储局部变量表、操作数栈、动态连接、方法出口等信息。每一个方法被调用直至执行完毕的过程,就对应着一个栈帧在虚拟机栈中从入到出栈的过程。当一个线程调用方法的时候,我们就会将这个方法放入到栈中(压栈),当调用多个方法的时候,最先压栈的方法会在栈底,最后压栈的方法会在栈顶,如下图所示。当这个方法使用完就会从这个栈里出去(弹栈)。

在《Java虚拟机规范中》,对这个内存区域规定俩类异常情况:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverFlowError异常。如果Java虚拟机栈容量可以动态扩展,当栈扩展是无法申请到足够的内存就会抛出OutOfMemoryError异常

3、本地方法栈(线程私有区域)

本地方法栈(Native Method Stacks)与虚拟机栈所发挥的作用是非常相似的,其区别只是虚拟机栈为虚拟机执行Java方式(也就是字节码)服务,而本地方法栈则是为虚拟机使用到的本地方法服务。与虚拟机栈一样,本地方法也会在栈深度溢出或者栈扩展失败的时候分别抛出StackOverFlowError和OutOfMemoryError异常。

4、Java堆(线程共享区域)

对于Java应用程序来说,Java堆(Java Heap)是虚拟机所管理的内存中最大的一块。Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,Java世界里“几乎”所有的对象是咧都在这里分配内存。在《Java虚拟机规范》中对Java堆的描述是:“所有的对象实例以及数组都应当在堆上分配“。

Java堆是垃圾收集器管理的内存区域,因此一些资料中它也被称为:GC堆Java堆实例分配的时,堆无法再扩展就会抛出OutOfMemoryError异常。

5、方法区(线程共享区域)

方法区(Method Area)与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类型信息、常量、静态变量、即时编译器编译后的代码缓存等数据。虽然《Java虚拟机规范》中把方法区描述为堆的一个逻辑部分,但是它却又一个别名叫做:“非堆”(Non-Heap),目的是与Java堆区分开来

说到方法区,不得不提一下“永久代”这个概念,尤其是在JDK8以前,许多Java程序员都习惯在HotSpot虚拟机上开发、部署程序,很多人都更愿意把方法区称呼为“永久代(Permanent Generation)”,或者俩者混为一谈。本质上俩者不能混为一谈,在JDK6的时候HotSpot团队就有发起永久代,逐步把原来采用本地内存(Native Memory)来实现方法区的计划了,到了JDK7的HotSpot,已经把原来发在永久代的字符串常量池、静态变量等移至Java堆中,而到了JDK8终于完全废弃了永久代的概念,改用本地内存中实现的元空间(Meta-space)来代替,把JDK7中永久代还剩余的内存全部移到元空间中。

根据《Java虚拟机规范》的规定,如果方法区无法满足新的内存分配需求时,将抛出OutOfMemoryError异常。

6、直接内存

直接内存(Direct Memory)并不是虚拟机运行时数据区的一部分,也不是《Java虚拟机规范》中定义的内存区域。但是这部分内存也被频繁的使用,而且也可能导致OutOfMemoryError异常。

在JDK1.4中心加入了NIO(new Input/Output)类,引入了一种基于通道(Channel)与缓冲区(Buffer)的I/O方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆里面的DirectByteBuffer对象作为这块内存的引用进行操作。这样能在一些场景中显著提高性能,因为避免了在Java堆和Native堆中来回复制数据。

7、运行时常量池

运行时常量池(Runtime Constant Pool)是方法区的一部分。Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池表(Constant Pool Table),用于存放编译期生成的各种字面量于符号引用,这部分内容将在类加载后存到方法区的运行时常量池中。这种特性被用得比较多的便是String类的intern()方法。

既然运行时常量池是方法区的一部分,自然受到方法区内存的限制,当常量池无法再申请到内存时会抛出OutOfMemoryError异常。

借道友法力一用:

========================== stay hungry stay foolish =============================

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值