JVM内存模型

原文:为知笔记外链

程序计数器(PC)

  1. 程序计数器(PC):一块较小的内存空间,可看作当前线程所执行的字节码的行号指示器。字节码解释器工作时通过改变该计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖该计数器完成。

  2. JVM的多线程是通过线程轮流切换并分配处理器执行时间的方式实现的,在任何一个确定的时刻,一个处理器都只会执行一条线程中的指令。因此,为了线程切换后能恢复到正确的位置,每天线程都需要有一个独立的PC,各个线程之间PC互不影响,独立存储,即该内存区域为“线程私有”。

  3. 若正在执行的是一个Java方法,则PC记录的是正在执行的虚拟机字节码指令的地址;如果是Native,则为空。

  4. 此内存区域是唯一一个JVM规范中没有规定任何OutOfMemoryError情况的内存区域。

Java虚拟机栈

  1. 线程私有,生命周期与线程相同。

  2. 虚拟机栈是Java方法执行的内存模型:每个方法在执行的同时会创建一个栈帧,用来存储 局部变量表、操作数栈、动态链接、方法出口等信息。

  3. 每个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。

  4. 局部变量表存放了编译期可知的各种基本数据类型、对象引用类型等。其中64位长度的long和double会占用2个局部变量空间(槽 Slot)。其余1个。因此,当进入一个方法时,这个方法需要在帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变局部变量表的大小。

  5. JVM规范中对该内存区域规定了两种异常状况:

  6. 如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;

  7. 如果虚拟机栈可以动态扩展(当前大部分都可以动态扩展,也允许固定长度),如果扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常。

本地方法栈

  • 虚拟机栈为虚拟机执行Java方法提供服务,本地方法栈为虚拟机中用到的Native方法提供服务。【StackOverflowError & OOM】

Java堆

  1. 被所有线程共享的一块内存区域,在虚拟机启动时创建。

  2. 用来存放对象实例。是GC管理的主要区域。

  3. 由于现在GC基本都采用分代收集算法,因此Java堆中还可以细分为:新生代【Eden控件,Survivor from、survivor to等空间】和老年代。

  4. 为了更好的回收、更快的分配内存,Java堆中可能划分出多个线程私有的分配缓冲区。

  5. Java堆可以出于物理上不连续的内存空间中,只要逻辑上连续即可。堆既可以是固定大小的,也可以是可扩展的。如果堆中没有内存完成实例分配,并且堆中无法再扩展,则会抛出OOM异常。

方法区

  • 各个线程共享的内存区域,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器后的代码等数据。不需连续内存、可固定可扩展,当方法区无法满足内存分配需求则会抛出OOM异常。

  • 运行时常量池,方法区的一部分。用来存放编译器生成的各种字面量和符号引用,该部分内容将在类加载后进入方法区的运行时常量池中存放。

对象的创建

遇到new指令时,首先会检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有则先执行对应的类加载过程。

对象内存分配算法:
  1. 指针碰撞(Bump the Pointer):假设Java堆中的内存是绝对规整的,所有用过的内存在一边,空闲的内存在另一边,中间放着一个指针作为分界点的指示器,则分配内存仅仅是把指针向空闲空间挪一段与对象大小相等的距离。
    修改指针并发问题解决方法:

    • CAS冲突重试功能

    • 把内存按照线程划分在不同的空间中进行,即每个线程在Java堆中预先分配一小块内存,成为本地线程分配缓冲区。

  2. 空闲列表(Free List):如果Java堆中的内存不是规整的,已使用的内存与空闲内存相互交错,则虚拟机必须维护一个列表记录哪些可用,在分配时从列表找出一块儿足够大的空间分配给对象实例并更新列表记录。

  3. 选择哪种分配方式由Java堆是否规整决定,而Java堆是否规整又由所采用的GC是否带有压缩整理功能决定。

对象的内存布局

分为 对象头、实例数据和对齐填充 3块区域。
  1. 对象头:包含两部分信息。第一部分用于存储对象自身的运行时数据【哈希码,GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID等】,长度为32bit/64bit【对应不同位的os】。另一部分是类型指针,jvm通过该指针确定该对象是那个类的实例。【如果对象是Java数组,则对象头中还必须有一块用来记录数组长度数据】。

  2. 实例数据是对象真正存储的有效信息,即代码中定义的各种类型的字段,包括父类中继承所得。

  3. 对齐填充不是必然,也无特别含义,起着占位符作用。因为jvm内存管理系统要求对象起始地址必须是8字节的整数倍,即对象的大小必须是8字节的整数倍。

对象的访问定位

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

推荐阅读更多精彩内容

  • 文章转自 http://blog.csdn.net/u012152619/article/details/4696...
    云狗狗狗狗狗阅读 604评论 1 4
  • 我们知道,计算机CPU和内存的交互是最频繁的,内存是我们的高速缓存区,用户磁盘和CPU的交互,而CPU运转速度越来...
    join_a922阅读 280评论 0 0
  • JVM的内存模型如下 主要分5个运行时内存,主要工作为: 1、程序计数器 程序计数器(Program Counte...
    amazingokc阅读 254评论 0 3
  • Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些区域都有各自不同的用途,...
    join_a922阅读 345评论 0 0
  • 首先,创建一个新项目。 我们找到入口文件index.ios.js,将下面代码替换进去。 从下往上理解这段代码,Ap...
    蜡小新阅读 226评论 0 0