JVM运行时内存区域

  Java虚拟机在执行Java程序时会把它所管理的内存划分为几块区域,分别用于负责不同的任务。有些区域的创建和销毁依赖于虚拟机进程,有的则依赖于用户线程的创建和销毁。废话不多说,我们来看看JVM运行时内存区域具体是个什么鬼,先看张图:


JVM运行时数据区

  原谅我不会画图,凑合着看看哈~
  我们从图上可以看到,Java虚拟机在运行程序时,并非所有的内存数据都同等对待,分为线程共享和线程私有的两种区域。线程共享即所有的线程都可以访问该区域的数据,而私有内存是伴随线程的创建而创建的区域,只对当前线程可见,并伴随线程的销毁而回收。
  我们接着来看两种区域下更细致的划分:

一、共享内存
  1. Java堆(Java Heap)
      在绝大多数情况下,Java堆算得上是JVM中空间最大的区域,因为这里的唯一作用就是存放程序中实例化的对象,几乎所有的对象实例都会在这里分配内存空间。它被所有线程共享,伴随着Java虚拟机的启动而创建。
      Java堆可以是物理上不连续的内存空间,但要求必须保证逻辑连续。在实现方面,可以设置为固定大小,也可以是可扩展的,主流的虚拟机是通过 -Mmx和 -Mms 进行配置实现。如果堆的剩余空间不足以分配实例对象需要的空间,且无法继续扩展,则会抛出OutOfMemoryError异常。
      那Java虚拟机怎么解决空间不足的问题呢?答案是内存回收 - 垃圾收集机制,即依赖垃圾收集器回收已经“死掉”的对象所占用的内存,以供后续新创建的对象使用。Java堆是垃圾收集器主要管理的区域,所以也叫做“GC堆”(Garbage Collected Heap)。目前主流的虚拟机实现收集器都采用分代收集算法,大致做法就是将Java堆细分为Eden区、From Survivor区、To Survivor区,新创建的对象优先会在Eden区分配空间,有的虚拟机还支持开启本地线程缓冲区(Thread Local Allocation Buffer. TLAB),(WTF,什么鬼玩意儿!)别慌,这个缓冲区只是为了帮助更快速地分配和回收内存,可以先不了解具体原理,但要知道如果开启了这个缓冲区,则优先在TLAB区域分配空间。当然一些例外的大对象,会直接在老年代分配内存空间,这么做是为了避免对象老年化的时候迁移大量数据。
      对象的回收算法还有标记-清除算法、复制算法、标记-整理算法,关于GC的收集算法,后续会单独介绍。
  2. 方法区(Method Area)
      方法区中存储的是已被虚拟机加载完毕的类信息、常量、静态变量以及即时编译器编译后的代码等数据。不同的虚拟机对方法区的实现方式各不相同,HotSpot虚拟机上采用永久代的方式来实现方法区,但这样容易遇到内存溢出的问题(永久代有 -XX:MaxPremSize的上限)。现在HotSpot虚拟机已经逐步采用Native Memory的方式来代替永久代,JDK1.7版本的HotSpot中,已经把原本放在永久代的字符串常量池移出。注意,这里指的是字符串常量池将不属于永久代,但仍属于方法区的一部分,即永久代并不等价于方法区。
      方法区和堆区一样,是所有线程共享的区域,并且Java虚拟机规范认为方法区是堆的一个逻辑部分,但实际上和堆是有区分的,它还有一个别名“Non-Heap”,即非堆。最大的不同之处是内存回收方面,Java虚拟机规范不要求方法区必须实现垃圾收集,这不代表方法区的数据会“长生不死”,方法区的回收目标主要是常量池的回收和类型的卸载,严格意义来讲,这些区域的内存回收也很有必要。
  3. 运行时常量池(Runtime Constant Pool)
      运行时常量池是方法区的一部分,它的作用是存放编译期生成的字面量和符号引用,这些数据来自于Class文件的常量池(Constant Pool Table),出这些之外,运行时常量池还会存放翻译后的直接引用。相较于Class文件的常量池,运行时常量池还具有动态性,这意味着其中存放的常量不仅仅是在编译期产生,还会在运行期存放新的常量。
二、线程私有内存
  1. Java虚拟机栈(Java Virtual Machine Stacks)
      虚拟机栈是对Java方法执行过程中的内存模型的描述。可以这么描述:每个方法在执行的时候会在虚拟机栈中创建一个对应的栈帧(Stack Frame)用于存储方法中的局部变量表、操作数栈、动态链接、方法出口等信息。该方法的调用、执行、完成的过程,对应着栈帧在虚拟机栈中的入栈到出栈的过程。局部变量表存放的数据有以下几种:
     a. 了编译期可知的各种基本数据类型(boolean、byte、char、short、int、float、long、double)
     b. 对象引用(reference,是一个指向对象起始地址的引用指针或者是指向代表对象的句柄)
     c. returnAddress类型(指向了一条字节码指令的地址)
    值得关注的是,局部变量表所需要的内存大小是在编译期确定并且完成分配,运行期不会动态改变局部变量表的大小。如果线程请求的栈深度大于虚拟机所允许的深度,会抛出StackOverFlowError异常;如果无法申请到足够的内存,则会抛出OutOfMemoryError异常。
  2. 本地方法栈(Native Method Stack)
      本地方法栈和虚拟机栈的作用几乎一致,但本地方法栈的服务对象是虚拟机所使用到的Native方法,而虚拟机栈是服务于虚拟机执行的Java方法。在Sun公司的HotSpot虚拟机的实现中,本地方法栈和虚拟机栈被放到了一起实现。
  3. 程序计数器(Program Counter Register)
      这是一块比较小的内存空间,它用于指定当前线程正在执行的字节码的行号,也叫行号指示器。基于这一点作用,就决定了程序计数器必须是线程私有的内存区域,因为在处理器的每个内核中,任意时刻都只会执行一个线程中的一个指令,只有保证程序计数器是线程独立的,线程之间的执行才不会因为处理器切换线程而互相影响。注意一点,如果当前线程正在执行Java方法,则计数器记录的正在执行的虚拟机字节码指令的地址;如果正在执行Native方法,则计数器值为空(Undifined)。
三、直接内存(Direct Memory)

  首先声明一点,直接内存并不属于Java虚拟机运行时内存区域。那这块区域是干什么的呢?它是JDK1.4引入NIO以后,基于通道(Channel)和缓冲区(Buffer)的I/O方式,使用Native函数库直接分配的堆外内存区域。在Java堆中会存储着一个对应的DirectByteBuffer对象作为这块堆外内存的引用,避免在Java堆和Native堆中来回复制数据。
  直接内存的大小不会收到Java堆的内存限制,但是会受到本机总内存大小和处理器寻址空间的限制。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343