深入理解java虚拟机(读书笔记一)

Java内存区域与内存溢出异常

运行时数据区域

java运行时数据区域包括:方法区、堆、虚拟机栈、本地方法区、程序计数器。
由线程所共享的区域:方法区和堆
线程隔离的数据区:虚拟机栈、本地方法区、程序计数器。

程序计数器
定义:当前线程所执行的字节码的行号指示器,字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。分支、循环、跳转、异常处理、线程恢复等基础功能都需要以来这个计数器来完成。
java多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的。在任何一个时刻,一个处理器都只会执行一条线程中的指令。如果线程正在执行java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果正在执行的是Native方法,这个计数器的值为空。

Java虚拟机栈
Java虚拟机栈是线程私有的,它的生命周期与线程相同。虚拟机栈描述的是方法执行的内存模型。每个方法在执行的同时,都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机中入栈到出栈的过程。
局部变量表存放了编译期可知的各种基本数据类型、对象引用、返回地址类型。除了long和double类型的数据会占用2个局部变量空间,其余数据类型只占用1个。局部变量表所需内存空间在编译期完成分配,在方法运行时不会改变局部变量表的大小。

StackOverflowError:线程请求的深度大于虚拟机所允许的深度时
OutOfMemoryError:内存不够时

本地方法栈
本地方法栈为虚拟机使用的Native方法服务,作用与java虚拟机栈基本相同,也会抛出以上两种异常。

Java堆
堆是虚拟机管理的内存中最大的一块,是被所有线程共享的区域,并在虚拟机启动时创建。此内存区域的唯一目的就是存放对象的实例,几乎所有的对象实例都在这里分配内存。java堆可以处于物理上不连续的区域,只要逻辑上连续即可。可扩展时可以通过-Xmx和-Xms控制。(OutofMemoryError)

方法区(别名Non-heap:非堆)
用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等。方法区中最重要的部分是运行时常量池。Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池,用于存放编译期生成的各种字面变量、符号引用、直接引用等,这些内容将在类加载后存放到方法区的运行时常量池中。

(OutOfMemoryError)。


直接内存

直接内存并不是java虚拟机运行时数据区的一部分,也不是java虚拟机规范中定义的内存区域。

NIO(new I/O)引入了一种基于通道(Channel)和缓冲区(Buffer)的I/O方式,可以使用操作系统本地方法库直接分配堆外内存,然后通过一个存储在java堆里面的DirectByteBuffer对象作为堆外直接内存的引用进行操作,避免了java堆内存和本地直接内存间的数据拷贝,可以显著提高性能。

虽然直接内存并不直接收到java虚拟机内存影响,但是如果java虚拟机各个内存区域总和大于物理内存限制,从而导致直接内存不足,动态扩展时也会抛出OutOfMemoryError异常。


java垃圾收集器重点关注的是java虚拟机的堆内存和方法区内存。





评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值