jvm学习笔记(一) java内存区域的划分

Java虚拟机在运行java程序的过程中会将它所管理的内存划分为若干个不同的区域,他们有各自不同的用途,创建以及销毁时间,如下图为java虚拟机运行时数据区:

1.程序计数器:他是一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器。如果线程正在执行的是一个java方法,这个计数器记录的就是正在执行的虚拟机字节码的指令地址。如果是native方法,则这个计数器的值为空。

2.java虚拟机栈:Java Virtual Machine Stacks,虚拟机栈是java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧用于存储局部变量表、操作数栈、动态连接、方法出口等信息。每个方法从调用到执行完成的过程就是一个栈帧在虚拟机中入栈到出栈的过程。这里的局部变量表存放了编译期可知的各种数据类型、对象引用以及returnAddress(指向了一条字节码指令的地址)。

3.本地方法栈:Native Method Stack,与虚拟机栈作用类似,区别就是虚拟机栈未虚拟机执行java方法(也就是字节码)服务,本地方法栈则是为虚拟机使用到的native方法服务。在HotSpot虚拟机中直接把虚拟机栈和本地方法栈合二为一了。

程序计数器、虚拟机栈和本地方法栈都是线程私有的,生命周期与线程相同,随着线程的创建而创建,随着线程的销毁而销毁。栈中的栈帧随着方法的进入和退出而进行着有条不紊的出栈入栈的操作。

4.java堆:java Heap,是java虚拟机所管理的内存中的最大的一块,这个内存的唯一目的就是存放对象实例,java堆是垃圾收集器管理的主要区域,所以也被称作“GC”堆。

5.方法区:Method Area,用于存储已被虚拟机加载的类信息,常量、静态变量、即时编译器编译后的代码等数据

6.运行时常量池:Runtime Constant Pool,它是方法区的一部分,用于存放编译期生成的各种字面量和符号引用,一般来说还会把翻译出来的直接引用也存储在运行时常量池中。运行时常量池具有动态性,常量并不一定在编译期才能产生,运行期间也可以将新的常量放入常量池中,比较常见的就是String类的intern()方法。

Java堆、方法区(运行时常量池)是所有线程共享的内存区域,随着虚拟机的启动而创建,随着虚拟机的关闭而销毁。

以上就是java虚拟机规范中定义的内存区域,除此之外还有一种内存叫直接内存(Direct Memory),它既不是虚拟机运行时数据区的一部分也不是java虚拟机规范中定义的内存区域。在jdk1.4中增加了NIO类(New Input/Output),引入了一种基于通道(Channel)与缓冲区(Buffer)的I/O方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在java堆中的DirectByteBuffer对象作为这块内存的引用进行操作,这样避免了java堆和Native堆中来回复制数据,在一些场景中显著的提高了性能。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值