java虚拟机的内存划分

本文介绍了Java虚拟机的内存划分,包括程序计数器、虚拟机栈、本地方法栈、堆内存和方法区(永久代)的作用和特点。详细讨论了各区域的生命周期、线程隔离性以及可能抛出的异常,如StackOverflowError和OutofMemoryError。此外,还提到了运行时常量池在不同JDK版本中的位置变化及其与字符串常量池的关系。
摘要由CSDN通过智能技术生成

java虚拟机在运行时,会把内存划分为数据区,不同的内存区域作用不同,生命周期不同。大致分为以下几个数据区:

上图来源于网络。这些区域的作用各不相同,大致说明各个区域作用:

程序计数器:主要是用于记录程序当前运行的位置,它控制具体执行哪一行代码。我们都知道,java代码在编译之后会成为字节码文件(.class),java虚拟机来执行字节码文件运行程序。由上图可知,程序计数器是线程隔离的数据区,也就是说,每一条线程都有各自不同的程序计数器。因为java虚拟机里的多线程是通过线程切换的方式轮流执行,在某一个时刻,一个处理器只会执行一条线程中的命令。所以为了在来回切换时能正确的找到执行位置,单独的线程会有单独的程序计数器,属于线程私有。另:如果线程执行的是Java方法,那么程序计数器记录的是字节码指令的地址;如果执行的是native方法,计数器值为空。这是虚拟机规定的唯一一个没有outOfMemaryError的内存区域。所以我觉得,不会内存溢出的原因是,计数器只存一个地址,不会太长所以不会有内存溢出的情况,在虚拟机中就不需要规定OurOfMemoryError。

java虚拟机栈:就是平时我们所说的栈内存,栈内存也是线程私有的,生命周期与线程相同。每个方法被执行时,会创建一个栈帧进栈,方法调用结束时出栈。一个方法调用的开始与结束就是一个栈帧入栈和出栈的过程,栈帧包括:局部变量表,操作栈,动态链接,方法出口。

       线程中不断调用方法的过程就是方法不断进栈的过程,也就是栈帧不断创建的过程,但是虚拟机所允许的栈内存是有大小的,如果内存超出了虚拟机所允许的最大内存,就会抛出java.lang.StackOverflowError异常。我刚刚调用了没有出口的递归,抛出了这个异常,方法被调用一万多次,但是修改该方法体之后,可调用次数就改变了,说明栈的大小确实是有限的,不同的方法占用内存大小不同,可调用次数也会改变。

       局部变量表所需的内存空间在编译时期完成分配,当进入一个方法时,这个方法需要在帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变变量的大小,所以,也就是说,变量的值可以改变但是变量的所占内存的大小是不可以改变的,所以我们在强制类型转换的时候,内存小的变量转为占内存大的变量是不会丢失精度的,但反之就不行了,多余的数据会被丢失掉,也就是说,当变量类型在编译时期一旦确定,不管转换成什么类型,占用大小就固定不再改变了。

本地方法栈:本地方法栈与java虚拟机栈不同的是,java虚拟机栈运行的是java方法,但是本地方法栈运行的是native方法。本地方法栈中运行的方法对使用语言,数据结构等并没有强制规定,并且也会抛出StackOverflowError和OurOfMemoryError异常。

java堆:堆内存是java虚拟机所管理内存中最大的一块,堆内存是被所有线程共享的内存区域,主要是用来放置对象的实例,在虚拟机启动时创建,数组以及几乎所有的对象实例都在堆上分配内存(随着JIT编译器和逃逸分析技术的发展,堆上分配对象不再成为绝对,对象也有可能不在堆内存分配)。垃圾收集器管理的主要区域就是堆内存,堆内存可以处在物理上不连续的内存空间中,只要逻辑上是连续的,在实现时可以是固定的,也可以是可扩展的(-Xmx和-Xms控制)。如果堆内存用完并且无法再扩展会抛出OurOfMemoryError。

方法区(永久代):线程共享的内存区域,用来存放虚拟机已经加载的类信息,常量,静态变量,即时编译器编译后的代码等数据(在Java编程语言和环境中,即时编译器是一个把Java的字节码(包括需要被解释的指令的程序)转换成可以直接发送给处理器的指令的程序)。它同样不需要连续的物理内存空间,可以选择固定大小或者可扩展。这个区域的内存回收主要是对常量池的回收和对类型的卸载,但是一般类加载之后很少去卸载。当方法区无法满足内存分配需求时,抛出OurOfMemoryError。

       方法区中包含运行时常量池。class文件中除了有类的版本,字段,方法,接口等,还有常量池,用于存放编译时期生成的各种字面量和符号引用。这部分内容在类加载之后存放在方法区的运行时常量池中。运行时常量池相对于class文件常量池具备动态性,运行时期也可以放入新的常量。这种特性被用的最多的是string的intern()方法。

    jdk1.6中字符串常量池存放在方法区中,当使用intern()方法时,查询字符串常量池是否存在当前字符串,若不存在则将当前字符串复制到字符串常量池中,并返回字符串常量池中的引用。

  jdk1.7中字符串常量池存放在堆中,当使用intern()方法时,先查询字符串常量池是否存在当前字符串,若字符串常量池中不存在则再从堆中查询,然后存储并返回相关引用;若都不存在则将当前字符串复制到字符串常量池中,并返回字符串常量池中的引用。

       当常量池无法满足内存分配需求时,抛出OurOfMemoryError。

 

 

 

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值