【深入理解java虚拟机】1. jvm内存分布

image

Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些区域有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而一直存在,有些区域则是依赖用户线程的启动和结束而建立和销毁。根据《Java虚拟机规范》的规定,Java虚拟机所管理的内存将会包括以下几个运行时数据区域。

image

1. 程序计数器

程序计数器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器。字节码解释器工作时,通过改变计数器的值来选取吓一跳需要执行的字节码指令。

线程私有

为了线程切换后可以正确恢复到争取的执行位置,每个线程都需要有单独的程序计数器。每个线程的程序计数器互不影响,独立存储。

所记录的值

如果线程执行的是java方法,那么计数器记录的是正在执行的字节码指令的地址

如果是本地方法,则应该为空。

不会发生OOM

2. 虚拟机栈

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

栈帧

每个方法都对应着一个栈帧, 栈帧存储着 局部变量表,操作数栈,动态链接,方法出口。

执行方法的过程 就对应着 栈帧 从虚拟机栈 入栈到出栈的过程。

操作数栈

public void test(){
    int a = 1;
    int b = 2;
    int c =  a+b;
}

0 iconst_1 // 1 入栈
1 istore_1 // 出栈赋值给局部变量表的a
2 iconst_2 // 2 入栈
3 istore_2 // 出栈赋值给局部变量表的b
4 iload_1  // 从局部变量表 把 1压入栈顶
5 iload_2  // 从局部变量表 把 2压入栈顶
6 iadd     // 1 + 2
7 istore_3 // 把3出栈 赋值给 变量表的c
8 return

局部变量表

存放了编译期可知的基本数据类型(boolean、byte、char、short、int、float、long、double)、对象引用 :可能是执行对象起始位置的指针 / 指向一个代表对象的句柄和returnAddress类型

存储空间

局部变量槽表示,64位长度的long和double类型 占用2个槽,其余的占1个槽

所需的内存空间在编译期间完成分配,运行时不会改变局部变量表的大小。

动态链接

每个栈帧都保存了 一个 可以指向当前方法所在类的 运行时常量池, 目的是: 当前方法中如果需要调用其他方法的时候, 能够从运行时常量池中找到对应的符号引用, 然后将符号引用转换为直接引用,然后就能直接调用对应方法, 这就是动态链接

方法出口

方法在哪返回

可能抛出的异常

StackOverflowError

如果线程请求的栈深度大于虚拟机所允许的深度

OutOfMemoryError

  1. 如果Java虚拟机栈容量可以动态扩展(HotSpot虚拟机不支持动态拓展),当栈扩展时无法申请到足够的内存会抛出OutOfMemoryError异常
  2. 不支持动态拓展的虚拟机空间申请成功了就不会oom,但是如果一开始申请栈空间 的时候就内存不足 也会oom

3. 本地方法栈

与虚拟机栈作用相似,为虚拟机使用到的本地(Native)方法服务

有的Java虚拟机(譬如Hot-Spot虚拟机)直接就把本地方法栈和虚拟机栈合二为一。

异常发生的情况与虚拟机栈一样

线程共享。存放java对象实例。

内存划分

内存划分方式与 虚拟机里的垃圾回收器有关。

采用分代回收算法的垃圾收集器 堆内存会分为新生代(Eden、Survivor)、老年代、永久代

堆所在的内存空间可以在物理上不连续,逻辑上应当连续。大对象 处于实现简单、存储高效的考虑 可能会要求连续的内存空间。

可拓展

-Xmx : 最小堆的大小, 虚拟机启动后,堆内存就这么大

-Xms : 最小堆占满后,进行GC,如果内存还是不够,拓展堆之 -Xms参数指定的大小

4. 方法区

线程共享,存储已被虚拟机加载的类型信息、常量、静态变量、即时编译器编译后的代码缓存。

永久代是方法区的一种基于分代回收 的实现。

Jdk7 hotspot 把 字符串常量池、静态变量 从永久代移除

Jdk8 用元空间代替。JDK 7中永久代还剩余的内容(主要是类型信息)全部移到元空间中。

OutOfMemoryError

如果方法区无法满足新的内存分配需求时OutOfMemoryError

运行时常量池

方法区的一部分

类的版本、字段、方法、接口等描述信息,还有常量池表(Constant Pool Table),用于存放编译期生成的各种字面量与符号引用,类加载后放到运行时常量池中。

动态性

并不一定只有编译期才能产生变量放入运行时常量池中,运行期间也可以将新的常量放入池中 ,比如String.intern()方法。

OutOfMemoryError

内存受到方法区内存的限制,当常量池无法再申请到内存时会抛出OutOfMemoryError异常

5. 直接内存

并不是虚拟机运行时数据区的一部分

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

显然,本机直接内存的分配不会受到Java堆大小的限制,但是,既然是内存,则肯定还是会受到本机总内存(包括物理内存、SWAP分区或者分页文件)大小以及处理器寻址空间的限制,一般服务器管理员配置虚拟机参数时,会根据实际内存去设置-Xmx等参数信息,但经常忽略掉直接内存,使得各个内存区域总和大于物理内存限制(包括物理的和操作系统级的限制),从而导致动态扩展时出现OutOfMemoryError异常。

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

推荐阅读更多精彩内容