前言
成都创新互联于2013年开始,是专业互联网技术服务公司,拥有项目成都网站设计、成都做网站网站策划,项目实施与项目整合能力。我们以让每一个梦想脱颖而出为使命,1280元龙山做网站,已为上家服务,为龙山各地企业和个人服务,联系电话:18980820575JAVA内存区域主要由程序计数器、java 虚拟机栈、本地方法栈、Java堆、方法区以及运行时常量池组成。本文将给大家详细介绍关于Java内存区域的划分与异常的相关内容,下面话不多说了,来一起看看详细的介绍吧。
运行时数据区域
JVM在运行Java程序时候会将内存划分为若干个不同的数据区域。
程序计数器
线程私有。可看作是 当前线程所执行的字节码的行号指示器 ,字节码解释器的工作是通过改变这个计数值来读取下一条要执行的字节码指令。
多线程是通过线程轮流切换并分配处理器执行时间来实现的,任何一个时刻,一个内核只能执行一条线程中的指令。 为了线程切换后能恢复到正确的执行位置,每条线程都需要一个独立的程序计数器 。这就是一开始说的“线程私有”。如果线程正在执行的方法是Java方法,计数器记录的是虚拟机字节码的指令地址;如果是Native方法,计数器值为空。 程序计数器是唯一一个在Java虚拟机规范中没有规定OOM(OutOfMemoryError)情况的区域 。
Java虚拟机栈
线程私有,生命周期和线程相同。Java虚拟机栈描述的是Java方法的内存模型:每个方法在执行时都会创建一个栈帧,存储 局部变量表、操作数栈、动态链接、方法出口信息 ,每一个方法从调用到结束,就对应这一个栈帧在虚拟机栈中的进栈和出栈过程。局部变量表保存了各种基本数据类型(int、double、char、byte等)、对象引用(不是对象本身)和returnAddress类型(指向了一条字节码地址)。
这部分区域可能发生两种异常:
本地方法栈
上述虚拟机栈为JVM执行Java方法服务,本地方法则为执行Native服务。其他和虚拟机栈类似,也会抛出StackOverflowError、OutOfMemoryError。
Java堆
常说的“栈内存”、“堆内存”,其中前者指的是虚拟机栈,后者说的就是Java堆了。 Java堆是被线程共享的 。在虚拟机启动时被创建。
Java堆的作用是存放对象实例,Java堆可以处于物理上不连续的内存空间中,只要求逻辑上连续即可。
方法区
线程共享的区域。存储已被虚拟机加载的类信息、常量、静态变量、即使编译器编译后的代码等数据。方法区无法满足内存分配需求时,抛出OutOfMemoryError。
运行时常量池
运行时常量池 是方法区的一部分 。C用于存放编译期生成的各种字面常量和符号引用,将在类加载后进入方法区的运行时常量池中存放。 Java语言不要求常量只能在编译期产生,换言之,在运行期间也能将新的常量放入 。
直接内存
直接内存不属于虚拟机运行时数据区的一部分,也不是内存区域。本机直接内存的分配不会受到Java堆的大小限制,但终究是内存,如果各个内存区域总和大于物理内存限制,还是会出现OutOfMemoryError。
对象的创建过程
虚拟机遇到一条"new"指令:
对象的内存布局
对象在内存中存储的布局可以分为3块区域:
对象头:存储对象自身的运行时数据,比如哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID等。另外还有一部分是类型指针, 即对象指向它的类元数据的指针,虚拟机通过该指针来确定这个对象属于哪个类的实例。
实例数据:对象真正有效的信息,在程序中定义的各种类型的字段内容;
对齐补充:非必须,占用符的作用。
对象的访问定位
Java程序通过栈上的引用来操作堆上的实例对象。比如
Person p = new Person();