资讯专栏INFORMATION COLUMN

jvm基础篇一之内存区域

Zachary / 1955人阅读

摘要:堆区堆是虚拟机所管理的内存中最大的一块,它是被所有线程共享的一块内存区域,该区域在虚拟机启动的时候创建。

运行时数据区域

   想要了解jvm,那对其内存分配管理的学习是必不可少的;java虚拟机在执行java程序的时候会把它所管理的内存划分成若干数据区域。这些区域有着不同的功能、用途、创建/销毁时间。java虚拟机所分配管理的内存区域如图1所示

程序计数器

   程序计数器是一块比较小的内存空间,它可以看做是当前线程所执行的字节码的执行位置的指针。在虚拟机中字节码,解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的指令;虚拟机完成分支、循环、跳转、异常处理、线程恢复等功能都需要依靠它。
   我们知道jvm多线程是通过线程的轮流切换并分配处理器执行时间的的方式来实现的,在任何时刻,一个处理器都只会执行一条线程中的指令。为了使线程被切换后能恢复到正确的执行位置,每条线程的程序计数器都应该是独立的,各条线程之间的计数器互不干涉,独立存储————程序计数器的内存区域为线程私有的内存。

   如果线程正在执行的是java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果执行的是Native方法,这个计数器的值则为空。此内存区域是唯一一个在jvm规范中没有规定任何OutOfMemoryerror情况的区域

java虚拟机栈

   java虚拟机栈为线程私有的内存,其生命周期与线程相同。每个方法在执行的时候会创建一个栈帧用于存储局部变量表、操作数栈、方法出口等信息。每一个方法从调用到执行完成,就对应着一个栈帧在虚拟机中从入栈到出栈的过程。其局部变量表存放了方法编译期可知的各种基本数据类型、对象引用、returnAddress类型(指向一条字节码指令的地址)jvm规范中,这个区域规定了两种异常状况:StackOverflowError和OutOfMemoryError。

本地方法栈

   本地方法栈的作用和虚拟机栈的作用很相似,它们的区别在于虚拟机栈为虚拟机执行java方法服务,而本地方法栈则为执行本地方法服务。有的虚拟机直接把本地方法栈和虚拟机栈二合一。与虚拟机栈一样,本地方法栈的异常也有两个:StackOverflowError和OutOfMemoryError。

java堆区

   java堆是虚拟机所管理的内存中最大的一块,它是被所有线程共享的一块内存区域,该区域在虚拟机启动的时候创建。这个区域的唯一目的就是存放对象实例。java堆是垃圾收集器工作的主要区域,由于垃圾收集器基本都采用分代收集的算法,所以java堆从垃圾收集器的角度来划分可以细分为新生代和老年代;从内存分配的角度来看,线程共享的java堆可能划分出多个线程私有的分配缓冲区。

   java堆区可以是物理上不连续的内存空间,只要逻辑上是连续的即可;一般而言我们的虚拟机java堆内存不是固定大小的,是可以扩展的。如果在堆中没有足够内存分配给对象实例,并且堆内存无法再扩展时,虚拟机将会抛出OutOfMemoryError异常。

方法区

   方法区与java堆区一样是各个线程共享的内存区域,这个区域存储了类信息、常量、静态变量等数据。java虚拟机规范中把方法区描述为堆得一部分逻辑,它又有一个名字——非堆,目的是与普通java堆进行区分。相对而言垃圾收集器在这个区域很少活动,因此一部分人把这个区域叫做“永久代”。这个区域的内存回收目标主要是针对常量池的回收和类型的卸载,然而类型卸载的条件是很苛刻的。该区域和和java堆区一样,当内存不够分配时会抛出OutOfMemoryError.

运行时常量池

   运行时常量池是方法区的一部分;一个Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是编译时常量池,用于存放编译期生成的常量。编译时常量池在类被加载后会放入方法区的运行时常量池中。与编译期常量池不同的是,运运行时常量池是动态的,运行期间产生的新的常量也会被放入这个区域,如:String类的intern()方法。

小结

   该篇对jvm内存只能算一个概览,给小伙伴们介绍了一些概念性的东西,很多地方是值得去深入研究的,比如具体一个对象实例是如何被分配到堆内存的,类的加载过程,方法执行时方法栈的入栈与出栈的具体过程······。jvm博大精深,我在这提供一个梗概,小伙伴们如果有时间可以细细推敲推敲。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/75499.html

相关文章

  • dubbo服务发布一之服务暴露

    摘要:整体流程以调试来演示服务的发布流程。暴露远程服务假如服务没有配置了属性,或者配置了但是值不是,就会执行远程暴露。封装了一个服务的相关信息,是一个服务可执行体。是一个服务域,他是引用和暴露的主要入口,它负责的生命周期管理。 整体流程以调试 om.alibaba.dubbo.demo.provider.DemoProvider来演示dubbo服务的发布流程。 1、启动Spring容器 参照...

    xialong 评论0 收藏0
  • JVM 内存模型

    摘要:是描述方法执行的内存模型每个方法执行的时候会同时创建一个栈帧,用于存储局部变量表操作数栈动态连接返回地址方法出口等信息。虚拟机是使用局部变量表完成参数值到参数变量表的传递过程。堆内存管理最大的一块。 showImg(https://segmentfault.com/img/bVLqsv?w=475&h=398); 1. 虚拟机栈 VM Stack 线程私有,生命周期与线程相同。VM S...

    SexySix 评论0 收藏0
  • JVM运行时数据区域

    摘要:虚拟机在执行程序的过程中会把它所管理的内存划分为若干个不同的数据区域,本篇文章将会对这些数据区域进行简略的介绍。运行时常量池运行时常量池是方法区的一部分。直接内存直接内存不是虚拟机运行时数据区的一部分,不是虚拟机规范中定义的内存区域。 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域,本篇文章将会对这些数据区域进行简略的介绍。JVM所管理的内存包括的...

    meislzhua 评论0 收藏0
  • jvm内存分配策略和性能监控

    摘要:概述本篇旨在讲清楚的内存分配策略,日志阅读,一些常见名词和提供的一些性能监控工具。内存分配与回收策略对象优先在分配大多数情况下,对象优先在新生代区中分配。当区域没有足够空间进行分配时,将发生一次。 概述 本篇旨在讲清楚jvm的内存分配策略,gc日志阅读,一些常见名词和jdk提供的一些性能监控工具。废话不多说,开始上货。 GC日志阅读 在开发的世界里,阅读日志是最基础的能力,也是解决问题...

    Baoyuan 评论0 收藏0
  • 深入理解jvm运行时区域

    摘要:内存区域虚拟机在运行程序时,会将其管理的内存区域划分成若干个不同的数据区域。运行时常量池运行时常量池是方法区的一部分。另外一部分官方称为用于存储自身运行时的数据,比如哈希值年龄锁状态标志偏向线程等。 前言 最近一直在看周志明老师的《深入理解虚拟机》,总是看了忘,忘了又看,陷入这样无休止的循环当中。抱着纸上得来终觉浅的想法,准备陆续的写几篇学习笔记,梳理知识的脉络并强化一下对知识的掌握。...

    ChanceWong 评论0 收藏0

发表评论

0条评论

Zachary

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<