Java内存区域

一 运行时数据区

1 运行时数据区包括方法区(Method Area)、堆(heap)、虚拟机栈(VM stack)、本地方法栈(Native Method stack)和程序计数器(program counter register)。

其中,方法区和堆是所有线程共享的数据区;虚拟机栈,本地方法栈和程序计数器是线程独占的数据区。


2 程序计数器

这是一块比较小的内存,虚拟机的概念模型中,字节码解释器工作时就是通过计数器的值选取下一条执行的指令。分支、循环、跳转、异常处理,线程恢复等基础功能都靠计数器实现。

程序计数器是Java虚拟机规范中唯一没有规定OutOfMemoryError情况的区域。


3 虚拟机栈

Java虚拟机栈的生命周期与其对应的线程相同。线程中的方法在执行时,会将一个栈帧压入虚拟机栈,栈帧中存储了局部变量表、操作数栈、动态链接、方法出口等信息。

如果线程请求的栈的深度大于虚拟机栈的最大限制,那么会抛出StackOverFlowError异常。如果栈深度可以动态扩展,当内存不够用时,会抛出OutOfMemoryError异常。


4 本地方法栈

本地方法栈与虚拟机栈功能类似。区别在于虚拟机栈为程序使用到的Java方法服务,本地方法栈为程序使用到的本地方法(Native Method)服务。

与虚拟机栈一样,它同样会抛出StackOverFlowError和OutOfMemoryError。


5 堆

堆的唯一目的,就是存储对象的实例。当内存空间不够时,将会抛出OutOfMemoryError异常。


6 方法区

存储加载进来的类信息,编译器编译后的代码,静态变量和常量等数据。

垃圾回收在方法区的工作,主要是对类型的卸载和对常量池的回收。

当空间无法满足存储需求时抛出OutOfMemoryError异常。


7 运行时常量池

运行时常量池是方法区的一部分。存储编译过程生成字面量、引用和其他常量。

会抛出OutOfMemoryError异常。


8 直接内存

直接内存不是Java运行时内存的一部分,但是和Java运行时内存相关。

JDK1.4中引入的NIO类(New Input/Output),引入了一种基于通道(Channel)与缓冲区(Buffer)的IO方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。这样避免了Java堆和Native堆之间来回复制数据,提高了性能。

空间不够时,抛出OutOfMemoryError异常。


二 Java对象的创建

1 首先去常量池中寻找类的符号引用,判断类是否被加载、解析和初始化。如果没有,则进行类加载。

2 对类的静态变量进行初始化。

3 为对象在堆中分配空间。

根据Java堆中空间是否规整(取决于垃圾回收器是否带有压缩整理功能),选取“指针碰撞”或者“空闲列表”的方式,为对象分配空间。

分配空间时会有线程安全的问题,解决这个问题的方案有两种:为空间分配增加同步处理,或者将内存空间分配的动作,按照线程分配到不同的空间进行

4 将分配的内存空间清零,初始化对象变量的默认值。

5 设置对象头中的信息(类元数据,对象哈希码,对象GC分代年龄)

6 初始化非静态变量(变量定义处)

7 执行构造器


三 对象的访问定位

目前常用的方式有两种:使用句柄和直接指针

1 使用句柄

好处是,引用指向了一个固定的句柄地址。当对象的位置发生改变时,只需要改变句柄中的地址。

2 直接指针

好处是,提高了对象访问的效率。因为其相对于使用句柄,减少了一次指针定位的操作。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值