JVM在运行java程序的过程中会把他所管理的内存划分为若干个不同的数据区域。

这些区域都有各自的用途和创建、销毁时间。有些区域随着虚拟机的启动而存在。有些区域则依赖用户线程的启动和结束而建立和销毁。依据《Java虚拟机规范1.7》规定,Java虚拟机所管理的内存分为下面几个区域:

程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区、运行时常量池、直接内存

程序计数器

是一块较小的内存空间。他能够看作是当前线程所运行的字节码的行号指示器。

在虚拟机的概念模型里,字节码解释器工作时就是通过改变这个计数器的值来选取下一条须要运行的字节码指令,分支,循环,跳转,异常处理,线程恢复等基础功能都依靠这个计数器来完毕。

因为Java虚拟机的多线程是通过线程轮流切换并分配处理器运行时间的方式来实现的,在不论什么一个确定的时刻,一个处理器(对于多核处理器仅仅有一个内核)都仅仅会运行一个线程中的指令。

因此,为了线程切换后能恢复到正确的运行位置,每条线程都须要有一个独立的程序计数器。各条线程之间计数器互不影响,独立存储。我们称这类内存区域为“线程私有”内存。

Java虚拟机栈

Java虚拟机栈也是线程私有的。他的生命周期与线程同样。虚拟机栈描写叙述的是Java方法运行的内存模型:每个方法在运行的同一时候都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出入口等。

每个方法从调用到运行完毕过程。就相应着一个栈帧在虚拟机中入栈到出栈的过程。

常常有人将Java内存分为堆内存(Heap)和栈内存(Stack),这样的分法是比較粗略的,java的内存区域远比这个复杂。

局部变量表存放了编译期可知的各种数据类型(boolean,byte,char,shot,int,float,long,double),对象引用类型(reference类型,他不同于对象本身,可能是一个指向对象起始地址的引用指针。也可能是一个代表对象的句柄或者其它与此对象关联的位置)和returnAddress类型(指向了一条字节码指令的地址)。

当中64位长度的lang和double类型的数据会占用2个局部变量空间(slot),其余数据类型仅仅占用1个。局部变量表所需的内存空间在编译期间完毕分配。当进入一个方法时。这种方法须要在帧中分配多大的局部变量空间是全然确定的。在方法运行期间不会改变局部变量表的大小。

在java虚拟机规范中,对于这个区域规定了两种异常状况:假设线程请求的栈深度大于虚拟机所同意的深度,将抛出StackOverflowException异常;假设虚拟机能够动态扩展。假设扩展到无法申请足够的内存。就会抛出OutMemoryError异常。

本地方法栈

虚拟机栈是为虚拟机运行Java方法也就是字节码服务的。而本地方法栈则为虚拟机用的Native方法服务。与虚拟机栈一样。本地方法栈也会抛出StackOverflowException和OutMemoryError异常。

Java堆

对于大多数应用来说。Java堆是Java虚拟机所管理的内存最大的一块,Java堆是被全部线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,差点儿全部的对象实例都要在这里分配内存,全部的对象实例以及数组都要在堆上分配,随着编译器的发展全部的对象都在对上分配变得不是那么绝对了。

Java堆是垃圾收集器管理的主要区域。因此非常多时候被称作GC堆。

从内存回收的角度来看:因为如今收集器基本都採用分代收集算法,所以Java堆还能够细分为:新生代和老年代。再仔细一点有Eden空间、From Survivor空间、To Survivor空间等。从内存的分配角度来看,线程共享的java堆可能划分出多个线程私有的分配缓冲区。

只是不管怎样划分都与存放内容无关。不管哪个区域存储的依旧是实例对象。进一步划分的目的是为了更好的回收内存,或者更快的回收内存。

依据Java虚拟机的规定,Java堆能够处于物理上不连续的内存空间,仅仅要逻辑上连续的就可以。

方法区

方法区与Java堆一样。是各个线程共享的区域,他用于存储已被虚拟机载入的类信息。常量,静态变量。即时编译器编译后的代码等数据。尽管java虚拟机规范把方法区描写叙述为堆的一个逻辑部分,可是他有别名——非堆,为了与java堆区分开来。

对于习惯在Hotspot虚拟机上开发、部署的开发人员来说,非常多人愿意把方法区称为永久代,本质上两者并不等价。

依据java虚拟机规范当方法区无法满足内存分配需求时将抛出OutOfMemoryError异常。

运行时常量池

是方法区的一部分,Class文件除了有类的版本号、字段、方法、接口等描写叙述信息外,另一项信息是常量池,用于存放编译期生成的各种字面量和符号引用,这部分内容将在类载入后进入方法区的运行时常量中存放。

他也是方法区的一部分,无法满足内存分配需求时将抛出OutOfMemoryError异常。

直接内存

不是虚拟机运行时数据区一部分,也不是java虚拟机规范中定义的内存区域。可是这部分内存也被频繁的使用,并且可能导致OutOfMemoryError异常。直接内存的分配不会受到java堆大小的限制,可是,既然是内存。肯定还是会受到本机内存的限制。运维在配置虚拟机參数时,会依据实际内存设置-Xmx等參数信息,但常常忽略直接内存。使得内存区域总和大于物理内存限制,从而导致动态扩展时出现OutOfMemoryError异常。

深入理解JVM:JVM执行时数据区域分类的更多相关文章

  1. JVM学习-运行时数据区域

    目录 前言 运行时数据区 程序计数器 Java虚拟机栈 局部变量表 基础数据类型 对象引用 returnAddress 操作数栈 动态链接 方法返回地址 Java堆 方法区 类型信息 字段描述符 方法 ...

  2. JVM<一>----------运行时数据区域

    参考:1.JVM Specification: http://docs.oracle.com/javase/specs/jvms/se7/html/jvms-2.html#jvms-2.5 2.< ...

  3. java内存结构(执行时数据区域)

    java虚拟机规范规定的java虚拟机内存事实上就是java虚拟机执行时数据区,其架构例如以下: 当中方法区和堆是由全部线程共享的数据区. Java虚拟机栈.本地方法栈和程序计数器是线程隔离的数据区. ...

  4. 深入理解Java虚拟机 -- 读书笔记(1):JVM运行时数据区域

    深入理解Java虚拟机 -- 读书笔记:JVM运行时数据区域 本文转载:http://blog.csdn.net/jubincn/article/details/8607790 本系列为<深入理 ...

  5. 深入理解Java虚拟机-JVM运行时数据区域

    一.运行时数据区域 1.程序计数器 程序计数器( Program Counter Register) 是一块较小的内存空间, 它可以看作是当前线程所执行的字节码的行号指示器. Java虚拟机的多线程是 ...

  6. 深入理解JAVA虚拟机之JVM性能篇---基础知识点(运行时数据区域)

    一. 运行数据区域划分 各个数据区域功能如下: 1. 程序计数器: 较小的一块内存空间,可以看做是当前线程所执行的字节码的行号指示器,每条线程都有一个独立的程序计数器,各条线程之间计数器互不影响,独立 ...

  7. JVM学习笔记:Java运行时数据区域

    JVM执行Java程序的过程中,会使用到各种数据区域,这些区域有各自的用途.创建和销毁时间.根据<Java虚拟机规范>,JVM包括下列几个运行时数据区域,如下图所示: 其中红色部分是线程私 ...

  8. JVM 内存区域 (运行时数据区域)

    JVM 内存区域 (运行时数据区域) 链接:https://www.jianshu.com/p/ec479baf4d06 运行时数据区域 Java 虚拟机在执行 Java 程序的过程中会把它所管理的内 ...

  9. JVM之基础概念(运行时数据区域、TLAB、逃逸分析、分层编译)

    运行时数据区域 JDK8 之前的内存布局 JDK8 之后的 JVM 内存布局 JDK8 之前,Hotspot 中方法区的实现是永久代(Perm),JDK8 开始使用元空间(Metaspace),以前永 ...

随机推荐

  1. 转 c++多线程编程

    c++多线程编程 一直对多线程编程这一块很陌生,决定花一点时间整理一下. os:ubuntu 10.04  c++ 1.最基础,进程同时创建5个线程,各自调用同一个函数 #include <io ...

  2. Django笔记:常见故障排除

    Django框架下MySQLdb模块在python3中无法使用的问题的解决方案 由于python3环境下目前还没有官方的mysqldb模块,Django框架中又强制要求使用mysqldb,为了解决这个 ...

  3. 洛谷 P1031 均分纸牌【交叉模拟】

    题目描述 有 N 堆纸牌,编号分别为 1,2,…, N.每堆上有若干张,但纸牌总数必为 N 的倍数.可以在任一堆上取若干张纸牌,然后移动. 移牌规则为:在编号为 1 堆上取的纸牌,只能移到编号为 2 ...

  4. Windows Server 2008 IE 无法调整安全级别

    开始”/“程序”/“管理工具”/“服务器管理器”命令,在弹出的服务器管理器窗口中,找到“安全信息”设置项,单击其中的“配置IE ESC”选项,打开如下图所示的IE增强安全配置窗口.

  5. Create Data Block Based On From Clause Query In Oracle Forms

    Example is given below to create a data block based on From Clause query in Oracle Forms. The follow ...

  6. dedecms 留言板中引用模板文件方法

    最近在做一个用dedecms搭建的网站,客户提出要有留言板,dedecms带了一个留言板的模块,安装倒是十分简便,但装完后发现界面十分粗糙.装修比较简单,但是发现遇到一个问题:网站通用的导航栏无法显示 ...

  7. 一个项目多个App项目搭建

    在testDjango项目中找到testDjango文件夹,打开urls.py路由配置文件并添加以下配置 from django.conf.urls import url,includefrom dj ...

  8. SparkStreaming和Drools结合的HelloWord版

    关于sparkStreaming的测试Drools框架结合版 package com.dinpay.bdp.rcp.service; import java.math.BigDecimal; impo ...

  9. 同步数据库到Codis代码

    同步mysql数据库到codis缓存中 public void syncRule() { // 根据时间戳获取Mycat中规则表数据 logger.info("start ..." ...

  10. Android 使用意图播放本地视频

    Android播放视频的方式有三种: 一.使用意图播放,调用本地安装的播放器,选择一个进行播放. 二.使用VideoView播放(VideoView事实上是对MediaPlayer的封装,使用起来非常 ...