Java虚拟机运行时数据区域

程序计数器(Program Counter)


程序计数器作为一个概念模型,这个是用来指示下一条需要执行的字节码指令在哪。

Java的多线程实际上是通过线程轮转做到的,如果是一个单核的机器(或单cpu),严格意义上在一个时间块中只会有一个线程在执行。为了线程切换以后能恢复到正确的执行位置,每个线程都需要有一个单独的计数器,每个计数器之间要是独立的互不干扰。

如果线程执行的是Java方法,那么PC指向的是正在执行的虚拟机字节码指令的区域,如果执行的是native方法,那么它是undefined。

Java虚拟机栈


Java virtue machine也是线程私有的,它拥有一个和线程相同的生命周期

虚拟机栈描述的是Java方法执行的内存模型;stack frame(栈帧)是一个经常谈及的概念,它用来储存内部变量表,操作数栈,动态链接,方法出口等等。

每一个方法从调用到执行完毕,也就对应着一个栈帧在虚拟机栈中的入栈和出栈

我们以前画图来说明内存区的时候,总是去关注Heap(堆内存)和stack(栈内存)这两部分,这是与对象内存分配最相关的两块内存区。通常所说的stack就是虚拟机栈,或者更具体的说是虚拟机栈中的局部变量表。

局部变量表存放了编译器可知的各种基本数据类型(boolean byte double char int short long float)对象引用(reference类型,并不是对象本身,可能是地址的引用指针,也可能是一个代代表对象的句柄)return address类型(指向一条字节码指令的地址)

局部变量表的意义就在于,可以把表所需的内存在编译器就进行分配,每次程序去调用一个方法的时候,方法需要在frame中分配多少的局部内存空间是确定的。

两种异常情况

如果线程请求的栈的深度大于虚拟机所允许的,就是StackOverFlowError,如果是支持动态拓展的虚拟机(大部分的现代虚拟机都支持)依然无法申请到足够的内存,就会报出OutOfMemoryError异常。

本地方法栈


本地方法栈是和Java虚拟机栈对应的一个概念,它们的作用也是相近的,唯一的不同是,本地方法栈执行的是native方法,而Java虚拟机栈执行的是Java方法(也就是字节码)服务

在Sun的HotSpot虚拟机里面,本地方法栈和虚拟机栈是一个。

Java堆


堆是被所有的线程所共享的一块区域,这块内存区域存在的唯一目的就是存放对象实例,在虚拟机启动的时候就会被创建,几乎所有的对象实例都会在这里被分配内存

所有的对象实例和数组都要在堆上分配 --《Java虚拟机规范》

随着JIT编译器的发展和逃逸技术的成熟,这句话也变得不是那么的绝对了。

GC(garbage collection)也发生在这个区域,所以有时候也被称为GC堆

方法区


方法区和Java堆相似,是线程共享的一段内存区域,它用于储存已经被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码。

听起来好像和Java堆很像,Java虚拟机标准里面也把它视为堆的一个逻辑部分,但是它被称作Non-Heap,目的是和Java堆区分开来。

Permanent Generation?那么,这个方法区就是永久代吗,并不是。只是在HotSpot虚拟机的设计中,用永久代来实现了方法区。(在JDK1.7中,已经把原本放在永久代的字符串常量池移出了)

运行时常量池(Runtime Constant Pool)

这也是方法区的一个较重要的部分,.class文件除了有类的版本,字段,方法,接口等描述信息外,还有一部分是常量池,用于在存放编译期生成的各种字面量(Literal)和符号引用(Symbolic References),这部分的内容在类加载以后进入运行时常量池中存放。

字面量比较好理解,是Java语言层面的常量,例如文本字符串,声明为final的变量

符号引用这个我第一时间没看懂什么意思,其实是编译原理的一个概念,包括以下的三种常量:

  • 类和接口的全限定名
  • 字段名称和描述符
  • 方法名称和描述符

动态性,这是运行时常量池的一个重要的特性,在运行期间也可以将新的常量放进常量区(包括基本包装类和String,也可以调用intern()将String强制放进常量池)

为什么需要运行时常量池呢?

  • 更少的内存。直接赋值的时候会利用常量池里面的对象,而不是去new了一个

  • 更快的速度 。‘==’比equals()更快

Integer a = 23;//在编译的时候会变成Integer i1=Integer.valueOf(40);使用的是线程池里面的对象

Integer b = new Integer(23);//创建了新的对象

ps.我感觉这个的设计思路和数据库连接池是差不多的,可以对照着去理解。

参考资料


《深入理解Java虚拟机》

关于Java虚拟机运行时数据区域的总结的更多相关文章

  1. Java虚拟机-运行时数据区域

    Java虚拟机管理的内存包括如图所示的运行时数据区域: 下面分别进行介绍: 1)程序计数器(Program Counter Register) 占用的内存空间比较小,主要作用就是标识当前线程执行的字节 ...

  2. Java虚拟机运行时数据区域划分

        Java虚拟机数据运行时区域 方法区(Method Area) 存储加载的类信息,常量,静态变量,编译器编译后的代码等数据.虽然JVM规范把方法区描述为堆的一个逻辑部分,但它却有一个别名叫做N ...

  3. Java虚拟机运行时数据区域及垃圾回收算法

    程序计数器 记录正在执行的虚拟机字节码指令的地址(如果正在执行的是本地方法则为空). Java 虚拟机栈 每个 Java 方法在执行的同时会创建一个栈帧用于存储局部变量表.操作数栈.动态链接.方法出口 ...

  4. JDK1.8-Java虚拟机运行时数据区域和HotSpot虚拟机的内存模型

    目录 介绍 官方文档规定的运行时数据区域 程序计数器 Java虚拟机栈 本地方法栈 虚拟机栈和本地方法栈溢出 Java堆 演示堆内存溢出 方法区 运行时常量池 演示方法区溢出 HotSpot虚拟机的内 ...

  5. 《深入理解Java虚拟机》(二)Java虚拟机运行时数据区

    Java虚拟机运行时数据区 详解 2.1 概述 本文参考的是周志明的 <深入理解Java虚拟机>第二章 ,为了整理思路,简单记录一下,方便后期查阅. 2.2 运行时数据区域 Java虚拟机 ...

  6. Java虚拟机运行时数据区

    运行时数据区程序计数器Java虚拟机栈本地方法栈Java堆(GC堆)方法区运行时常量池 运行时数据区 Java虚拟机在运行Java程序时,会将它所管理的内存划分为若干个内存区域.这些数据区域有各自的用 ...

  7. 笔记:Java虚拟机运行时数据区

    Java虚拟机在执行Java程序的过程中会把它管的内存划分为以下若干个不同的区域: 1.程序计数器 程序计数器是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器:由于Java虚拟机的 ...

  8. Java虚拟机运行时内存区域简析

    figure:first-child { margin-top: -20px; } #write ol, #write ul { position: relative; } img { max-wid ...

  9. 深入理解Java虚拟机&amp;运行时数据区

      其中,程序计数器.虚拟机栈.本地方法栈3个区域随线程而生,随线程而灭.

  10. [jvm]运行时数据区域详解

    了解虚拟机是怎么使用内存的,有助于我们解决和排查内存泄漏和溢出方面的问题.详解java虚拟机内存的各个区域,分析这些区域的作用服务对象以及可能发生的问题. 一.运行时数据区域 java虚拟机在执行ja ...

随机推荐

  1. PYTHON 写函数,检查传入字典的每一个value的长度,如果大于2,那么仅保留前两个长度的内容,并将新内容返回给调用者

    def a4(arg): ret = {} for key,value in arg.items(): if len(value) > 2: ret[key] = value[0:2] else ...

  2. TCP的粘包现象

    看面经时,看到有面试官问TCP的粘包问题.想起来研一做购物车处理数据更新时遇到粘包问题,就总结一下吧. 1 什么是粘包现象 TCP粘包是指发送方发送的若干包数据到接收方接收时粘成一包,从接收缓冲区看, ...

  3. 【Hadoop】HDFS的运行原理

    博文已转移,请借一步说话http://www.weixuehao.com/archives/596 简介 HDFS(Hadoop Distributed File System )Hadoop分布式文 ...

  4. xavante运行cgilua流程

    安装软件 需要安装三个lua 库:  xavante wsapi cgilua luarocks install xavante http://keplerproject.github.io/xava ...

  5. js闭包之初步理解( JavaScript closure)

    闭包一直是js中一个比较难于理解的东西,而平时用途又非常多,因此不得不对闭包进行必要的理解,现在来说说我对js闭包的理解. 要理解闭包,肯定是要先了解js的一个重要特性, 回想一下,那就是函数作用域, ...

  6. 通过SCVMM分配iSCSI存储

    除了使用基于SMB3.0应用程序的文件共享外,还可以使用iSCSI目标服务器的SAN存储,然后在SCVMM控制台中添加基于SMI-S类型的存储,步骤如下: 1.将一台安装了 iSCSI目标 功能的Wi ...

  7. MVC5-Scaffolder

    [转]Visual Studio.net 2013 asp.net MVC 5 Scaffolding代码生成向导开源项目 提高开发效率,规范代码编写,最好的方式就是使用简单的设计模式(MVC , R ...

  8. Data visualization 课程 笔记3

    Learn how humans work to create a more effective computer interface 三种reasoning的方式  Deductive Reason ...

  9. 【运维工程师必知必会】——MySql基础

    一.SQL语句 1.分类 DDL(data definition language)数据定义语言(create.alter.drop),管理基础数据.例如:库.表     #运维要熟练,开发也要熟练 ...

  10. .net 多线程之线程取消

    //线程取消不是操作线程,而是操作信号量(共享变量,多个线程都能访问到的东西,变量/数据库的数据/硬盘数据) //每个线程在执行的过程中,经常去查看下这个信号量,然后自己结束自己 //线程不能别人终止 ...