java虚拟机结构(概览二)

java虚拟机结构(概览二)

1、运行时数据区

java虚拟机定义了若干种程序运行期间会使用到的运行时数据区,其中有一些会随着虚拟机的启动而创建,随着虚拟机退出而销毁。另外一些则是与线程一一对应的,这些与线程对应的数据区域会随着线程开始和结束而创建和销毁。

1.1、pc寄存器

java虚拟机可以支持多条线程同时执行,每一条java虚拟机线程都有自己的pc(program counter)寄存器。任意时刻,一条java虚拟机线程会执行一个方法的代码,这个正在被执行的方法称为该线程的当前方法(current method)。这个方法是非native的,那pc寄存器就保存java虚拟机正在执行的字节码指令的地址,如果该方法是native的,那pc寄存器的值是undefined。

1.2、java虚拟机栈

每一条java虚拟机线程都有自己私有的java虚拟机栈(java virtual machine stack),这个栈与线程共同创建,用于存储栈帧(frame)。java虚拟机栈的作用与传统语言(例如C语言)中的栈非常类似,用于存储局部变量与一些尚未算好的结果。另外,它在方法调用和返回中也扮演了很重要的角色。因为除了栈帧的出栈和入栈之外,java虚拟机不会在受其他因素的影响,所以栈帧可以在堆中分配,java虚拟机所使用的内存不需要保证是连续的。 java虚拟机规范既允许java虚拟机被实现成固定大小,也允许根据计算动态来扩展和收缩。如果采用固定大小的java虚拟机栈,那每一个线程的java虚拟机栈容量可以在线程创建的时候独立选定。 java虚拟机栈可能发生如下异常情况:

  如果线程请求分配的栈容量超过java虚拟机允许的最大容量,java虚拟机将会抛出一个StackOverflowError异常。
  如果java虚拟机栈可以动态扩展,并且在尝试扩展的时候无法申请到足够的内存,或者在创建新的线程时没有足够的内存去创建对应的虚拟机栈,那java虚拟机将会抛出一个OutOfMemoryError异常。

1.3、java堆

在java虚拟机中,堆(heap)是可供各个线程共享的运行时内存区域,也是供所有类实例和数组对象分配内存的区域。 java堆在虚拟机启动的时候就被创建,它存储了被自动内存管理系统(automatic storage management system,也就是常说的garbage collector垃圾收集器)所管理的各种对象,这些受管理的对象无需也无法显式地销毁。java堆的容量可以是固定的,也可以随着程序执行的需求动态扩展,并在不需要过多空间时自动收缩。java对所使用的内存不需要保证是连续的。 java堆可能发生如下异常情况:

  如果实际所需要的堆超过了自动内存管理系统能提供的最大容量,那java虚拟机将会抛出一个OutOfMemoryError异常。

1.4、方法区

在java虚拟机中,方法区(method area)是可供各个线程共享的运行时内存区域。方法区与传统语言中的编译代码存储区(storage area for compiled code)或者操作系统进程的正文段(text segment)的作用非常类似,它存储了每一个类的结构信息,例如,运行时常量池(runtime constant pool)、字段和方法数据、构造函数和普通方法的字节码内容,还包括一下在类、实例、接口初始化时用到的特殊方法。 方法区在虚拟机启动的时候创建,虽然方法区是堆的逻辑组成部分,但是简单的虚拟机实现可以选择在这个区域不实现垃圾收集与压缩。方法区的容量可以是固定的,也可以随着程序执行的需求动态扩展,并在不需要过多空间时自动收缩。方法区在实际内存空间中可以是是不连续的。 方法区可能发生如下异常情况:

  如果方法区的内存空间不能满足内存分配请求,那java虚拟机将会抛出一个OutOfMemoryError异常。

1.5、运行时常量池

运行时常量池(runtime constant pool)是class文件中每一个类或接口的常量池表的运行时表现形式,它包含若干种不同的常量,从编译期可知的数值字面量到必须在运行期解析后才能获得的方法或字段引用。运行时常量池类似于传统语言中的符号表(symbol table),不过它存储数据的范围比通常意义上的符号表更为广泛。 每一个运行时常量池都在java虚拟机的方法区中分配,在加载类和接口到虚拟机后,就创建对应的运行时常量池。 在创建类和接口的运行时常量池时,可能发生如下异常情况:

  当创建类和接口是,如果构造运行时常量池所需要的内存空间超过了方法区所能提供的最大值,那java虚拟机将会抛出一个OutOfMemoryError异常。

1.6本地方法栈

java虚拟机实现可能会使用到传统的栈(通常称为C stack)来支持native方法(指使用java以外的其他语言编写的方法)的执行,这个栈就是本地方法栈(native method stack)。当java虚拟机使用其他语言(例如C语言)来实现指令集解释器时,也可以使用本地方法栈。如果java虚拟机不支持native方法,或是本身不依赖传统栈,那么可以不提供本地方法栈,如果支持本地方法栈,那这个栈一般会在线程创建的时候按线程分配。 本地方法栈可能发生如下异常情况:

  如果线程请求分配的栈容量超本地方法栈允许的最大容量,java虚拟机将会抛出一个StackOverflowError异常。
  如果本地方法栈可以动态扩展,并且在尝试扩展的时候无法申请到足够的内存,或者在创建新的线程时没有足够的内存去创建对应的本地方法栈,那java虚拟机将会抛出一个OutOfMemoryError异常。

未完待续~

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值