深入理解Java虚拟机 - 虚拟机内存划分

在内存管理方面,Java相对于C和C++的区别在于Java具有内存动态分配以及垃圾收集技术,但平时我们很少去关注JVM的内存结构以及GC,在出现内存泄露或溢出方面的问题,排查工作将变得异常艰难。

1. 运行时数据区域

Java虚拟机在执行Java程序时会将其管理的内存按照用于划分为若干个不同的数据区域,这些区域有着各自不同的生命周期。根据《JAVA虚拟机规范》,Java虚拟机管理的内存会包含以下几个区域。其中可以分为共享内存区以及线程隔离数据区两个部分。

      

2. 程序计数器

程序计数器是一个非常小的内存空间,可以看做当前线程执行的字节码的行号指示器。在虚拟机中,程序的执行,跳转,循环等都需要该计数器的值来选取下一条要执行的指令。程序计数器占用的内存空间非常小,而且通常不会出现OOM或SOF等。

但需要注意的是,程序计数器如果正在执行一个Java方法,则其中保存的是下一条指令的地址;如果执行的是Natvie方法,则为空。

3. JAVA 虚拟机栈

同程序计数器一样,虚拟机栈也是线程私有的,其周期与线程的生命周期相同,其描述的是Java方法执行的内存模型:每个方法执行的时候都会同时创建一个栈帧用于存储局部变量表、操作栈、动态链接、方法出口等信息。每一个方法被调用直到执行完成的过程就对应着一个虚拟机栈从入栈到出栈的过程。

Java内存区通常习惯被分为堆和栈,这种划分方法实际上是比较粗略的,此时所谓的栈就是虚拟机栈。

局部变量表存放了编译期可知的基本数据类型,对象引用和returnAddress类型。

需要注意的是,局部变量表所需的内存空间在编译期内完成分配,当进入一个方法时,这个方法在帧中分配多大的内存空间是完全确定的,在运行时也不会改变。

在JAVA虚拟机规范中,定义了两种异常:如果线程请求深度大于虚拟机允许深度,则会抛出StackOverflowError异常;如果扩展无法申请到足够内存时会抛出OutOfMemoryError异常。

由-Xss设定栈大小

// 虚拟机栈和本地方法栈的大小 = 线程允许最大内存 - 最大堆容量 - 最大方法区容量
// 在多线程时,给每个线程分配的栈越大,越容易出现异常

4. 本地方法栈

本地方法栈只为虚拟机使用的Native方法服务,其他类似于虚拟机栈

5. 堆

Java Heap是java内存管理中最大的一部分,Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象的实例。

Java Heap是GC的主要区域,有时候会被成为“GC堆”。Java堆还可以进行细分:新生代和老年代:再细致一点可以划分为Eden空间、From Survivor空间、To Survivor空间等。虽然被划分为多个空间,其存放的仍是对象,目的是为了更好更快的回收内存。

Java堆可以位于不连续的内存空间上,并且可以通过配置进行扩展 -Xmx 和-Xms控制,如果在堆中没有完成内存分配,而且也无法扩展时,会抛出"OutOfMemoryError“异常。

示例:

   

// 堆溢出
// -Xmx 堆最大值
// -Xms 堆最小值
// -XX:+HeapDumpOnOutOfMemoryError 设置虚拟机在异常时转储当前内存堆
// 可以使用Eclipse Memory Analyzer 对转储的堆进行分析

6. 方法区

方法区也是线程共享的内存区域,用于存储加载的类,常量,静态变量和即时编译器编译后的代码。

对于使用HotSpot虚拟机的开发人员来说,其通常将其称为永久代,意为方法区内较少出现垃圾收集,主要是对常量的清理和类的卸载,但回收的内存的效果较其他内存区域要差。

7. 运行时常量池

Runtime constant pool是方法区的一部分,其中会保存在class文件中的字面量和符号引用,常量池在运行时也可以将数据导入

8. 直接内存

直接内存并非虚拟机运行内存的一部分,是属于运行环境的内存区域。

如JAVA引入的NIO,可以直接操作本机内存,虽然不再受JVM堆大小的限制,但仍然受运行环境内存限制,当超限时,也会抛出OOM异常。

9. 对象访问

Object obj = new Object()

Object obj 将会反映在Java栈的本地变量表中,作为一个reference类型数据出现。

new Object 将会反应在堆中,并分配一块堆的结构化内存

java访问对象可能采用直接访问后者句柄访问两种方式,其中直接访问方式速度快,句柄访问稍慢,但句柄中存放稳定的句柄地址。

时间: 2024-10-25 02:14:28

深入理解Java虚拟机 - 虚拟机内存划分的相关文章

Java虚拟机的内存划分

内存概述 内存是计算机中的重要原件,临时存储区域,作用是运行程序.我们编写的程序是存放在硬盘中的,在硬盘中的程序是不会运行的,必须放进内存中才能运行,运行完毕后会清空内存.Java虚拟机要运行程序,必须要对内存进行空间的分配和管理.. Java虚拟机的内存划分 为了提高运算效率,就对空间进行了不同区域的划分,因为每一片区域都有特定的处理数据方式和内存管理方式 JVM的内存划分: 栈: 存储局部变量 堆:存储new出来的数组或对象 一个数组内存图 原文地址:https://www.cnblogs.

Java中的内存划分

Java程序在运行时,需要在内存中分配空间.为了提高运行效率,就对数据进行了不同的空间划分.因为每一片区域都有特定的数据处理方式和内存管理方式. 具体分为5种内存空间: 程序计数器:保证线程切换后能恢复到原来的执行位置. 虚拟机栈:(栈内存)为虚拟机执行java方法服务,方法被调用时,创建栈帧- 本地方法栈:为虚拟机执行使用到的Native方法服务 堆内存:存放所有new出来的东西 方法区:存储被虚拟机加载的类信息,常量,静态常量,静态方法等. 运行时常量池(方法区的一部分) GC对他们的回收:

Java虚拟机:内存模型详解

版权声明:本文为博主原创文章,转载请注明出处,欢迎交流学习! 我们都知道,当虚拟机执行Java代码的时候,首先要把字节码文件加载到内存,那么这些类的信息都存放在内存中的哪个区域呢?当我们创建一个对象实例的时候,虚拟机要为对象分配内存,Java虚拟机又是如何配分内存的呢?这些都涉及到Java虚拟机的内存划分机制,今天我们就来探究一下Java虚拟机的内存模型. Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域,这些区域都有各自的用途以及创建和销毁的时间,有的区域随

Java虚拟机的内存结构

我们都知道虚拟机的内存划分了多个区域,并不是一张大饼.那么为什么要划分为多块区域呢,直接搞一块区域,所有用到内存的地方都往这块区域里扔不就行了,岂不痛快.是的,如果不进行区域划分,扔的时候确实痛快,可用的时候再去找怎么办呢,这就引入了第一个问题,分类管理,类似于衣柜,系统磁盘等等,为了方便查找,我们会进行分区分类.另外如果不进行分区,内存用尽了怎么办呢?这里就引入了内存划分的第二个原因,就是为了方便内存的回收.如果不分,回收内存需要全部内存扫描,那就慢死了,内存根据不同的使用功能分成不同的区域,

深入理解java虚拟机系列(一):java内存区域与内存溢出异常

文章主要是阅读<深入理解java虚拟机:JVM高级特性与最佳实践>第二章:Java内存区域与内存溢出异常 的一些笔记以及概括. 好了开始.如果有什么错误或者遗漏,欢迎指出. 一.概述 先上一张图 这张图主要列出了Java虚拟机管理的内存的几个区域. 常有人把Java内存区分为堆内存(Heap)和栈内存(Stack),这种分法比较粗糙,Java内存区域的划分实际上远比这复杂,从上图就可以看出了.堆栈分法中所指的"栈"实际上只是虚拟机栈,或者说是虚拟机栈中的局部变量表部分.接下

深入理解java虚拟机(二)HotSpot Java对象创建,内存布局以及访问方式

内存中对象的创建.对象的结构以及访问方式. 一.对象的创建 在语言层面上,对象的创建只不过是一个new关键字而已,那么在虚拟机中又是一个怎样的过程呢? (一)判断类是否加载.虚拟机遇到一条new指令的时候,首先会检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号代表的类是否被加载.解析并初始化.如果没有完成这个过程,则必须执行相应类的加载. (二)在堆上为对象分配空间.对象需要的空间大小在类加载完成后便能确定.之后便是在堆上为该对象分配固定大小的空间.分配的方式也有两种:

《深入理解Java虚拟机》读书笔记---第二章 Java内存区域与内存溢出异常

Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来.这一章就是给大家介绍Java虚拟机内存的各个区域,讲解这些区域的作用,服务对象以及其中可能产生的问题. 1.运行时数据区域 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域. 1.1程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可以看作是当前线程所执行的字节码的行号指示器.在虚拟机的概念模型中里,字

JVM内存结构---《深入理解Java虚拟机》学习总结

本文转自:http://www.cnblogs.com/jilodream/,作者:王若伊_恩赐解脱 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域的用途各不相同,同时也依据着各自的执行规则,独立的创建和销毁数据. 虚拟机内存的划分,如图所示: 线程之间互相独立的区域有: 虚拟机栈 .本地方法栈.程序计数器 线程可以共享数据的区域: 方法区 .堆 每个区域的作用分别如下: 程序计数器 Program Counter Register: 众所周知,虚

深度理解java虚拟机读书笔记(二)HotSpot Java对象创建,内存布局以及访问方式

内存中对象的创建.对象的结构以及访问方式. 一.对象的创建 在语言层面上,对象的创建只不过是一个new关键字而已,那么在虚拟机中又是一个怎样的过程呢? (一)判断类是否加载.虚拟机遇到一条new指令的时候,首先会检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号代表的类是否被加载.解析并初始化.如果没有完成这个过程,则必须执行相应类的加载. (二)在堆上为对象分配空间.对象需要的空间大小在类加载完成后便能确定.之后便是在堆上为该对象分配固定大小的空间.分配的方式也有两种:

《深入理解Java虚拟机》 -- 内存

JVM对于操作系统来说是一种应用程序,JVM要运行的时候,操作系统会创建对应的进程而且分配一定大小的内存. 一.内存结构 当虚拟机得到系统分配的内存后,它在其内存空间中就是老大,管理对象内存的分配以及对象内存的回收,同时可以根据虚拟机的规范对其内存空间划分不同的区域.主要分为运行数据区.执行引擎.本地接口与本地类库.结构如下图: 大多数程序员更多关注的是变量是怎么存储的或者存在哪个区域.还有对象的内存分配关系,所以在这个层面上来看可以分为堆与栈两个内存区域,而这两个区域都是在运行数据区,所以我们