JVM的内存机构

1.什么是JVM

首先要说明什么是JVM。

JVM是java虚拟机的简称。它负责将编译生成的字节码文件(.class)解释并执行。JVM的可以在不同操作系统上运行,也是Java跨平台特性的原因之一。

需要注意的一点是,JVM虽然属于Java语言体系,但实际上具有多种JVM虚拟机实现。更具不同的操作系统,JVM底层的实现方式也有所不同。目前最为主流的Java虚拟机有HotSpot、JRockit等。


2.JVM的内存结构

JVM的内存结构,主要研究其运行时数据区的内存分配和管理情况。其构成如下图


JVM的内存机构_第1张图片
JVM结构

首先类装载器classLoader负责加载.class文件进入JVM。


程序计数器(Register)

这是一块很小的内存区域

功能:负责执行字节码(.class)。这些字节码是直接操作计算机的指令,JVM中的字节码解释器,就是通过改变程序计数器的值,实现顺序、分支、循环等最基本操作的。可以理解为单片机中的寄存器。

线程可见性:该区域是线程私有的。每个线程都有自己独立的程序计数器空间。(注意,Java虚拟机是多线程操作的,通过分配时间调度各个线程。且在同一时刻,多核处理器中的每一核都只执行自己的线程的中的指令)


虚拟机栈(stack)

通常程序员也习惯称之为Java栈,或方法栈。

功能:描述Java方法执行的内存区域。每个方法在执行时都会在该栈中创建一个栈帧(可以理解为一个元素),其中存储着局部变量表,操作栈,出口信息等。每个方法从开始执行到执行结束,实际就是一个进栈——出栈的过程。

说明:局部变量表,里面存放着java的8中基本数据类型和引用类型和returnAddress类型。基本数据类型不再多说,除了64位的long和double使用两个局部变量空间,其余都使用一个。引用类型,通过指针或者句柄(其实就是个编号,可以指向一个对象)指向堆中的Java对象。局部变量表在创建时,大小就确定,不可更改。

线程可见性:栈同样是线程私有的,且声明周期与线程相同。


Java堆(heap)

Java对象存放的空间,空间较大。(通常也习惯简单的把JVM内存结构分为堆和栈两部分)

功能: 管理Java对象的生命周期,所有的Java对象实例都存放在这块内存中。(深入的考虑,其实还有一部分对象不由它管理)

说明:该区域是GC垃圾回收机制主要执行的地方,根据垃圾回收机制,该区域又可分为新生代、老生代、永生代,各个部分具有不同的回收算法,先不做详细介绍。

该区域的大小可以通过-Xms 和-Xmx进行手动分配。默认Xms分配初始最小运行内存,为计算机物理内存的1/64,Xmx表示最大内存,默认为物理内存的1/4

线程可见性:所有线程共享,且堆内存不要求物理上的连续内存,而是逻辑上的连续。


本地方法栈

功能与虚拟机栈相似,只不过它负责存放和执行的是本地方法。

关于本地方法,简单理解就是不是使用Java语言写的方法,这部分方法使用的语言,取决于JVM运行的操作系统平台。通常来说,使用的是C语言辨析本地方法。此外本地方法接口的存在,可以使java调用非Java语言,这通常用来在不同的OS上实现不同的JVM。例如JNI可以调用C语言的库函数。

这一特性更可以实现Java代码同OS的隔离,保证跨平台性。


方法区

功能:存储着加载后的类的信息,相关常量,static静态量等。

线程可见性:各个线程共有。

说明:在部分虚拟实现中,由于常量可以被理解为创建出来就可以不会轻易变化和销毁的(注意,不是不销毁,只是很难销毁且没有销毁的必要),就直接选择将方法区实现在永生代中。这样GC的垃圾回收就可以扩展到这个区域。


运行时常量池

上图中并没有这个区域,它存在于方法区中。

首先要说明,这里说的常量不仅是Java语言定义的常量,不改变的量。而是指代码编译后,.class中的符号引用(用于指向本Java程序调用的外部Java类库)和普通Java常量的集合。二者统称为运行时常量,存放在常量池中。

此外,运行时常量池具备动态性,在Java运行时,仍然可以动态的向其中添加常量。String类的intern方法就利用这个特性。

你可能感兴趣的:(JVM的内存机构)