《深入理解java虚拟机》笔记(1)运行时数据区域

1、Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的“高墙”,墙外面的人想进去,墙里面的人却想出来。

2、运行时数据区域划分

  java虚拟机在执行java程序的过程中会把它所管理的内存划分为若干个区域,这些区域都有各自的用途,创建和销毁时间,有的区域随着虚拟机进程的启动而存在,有的区域则依赖用户线程的启动和结束而建立和销毁,根据《Java虚拟机规范(Java SE 7版)》的规定,java虚拟机分为以下区域。

  

  2.1、程序计数器(Program Counter Register)

  程序计数器属于线程私有,是一块较小的空间,可以看作是当前线程所执行的字节码的行号指示器。字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。每条线程都有独立的计数器,各条线程之间计数器互不影响,独立存储。此内存区域是唯一一个在java虚拟机规范中没有规定任何OutOfMemoryError情况的区域。

  2.2、java虚拟机栈(VM Stack)

  和程序计数器一样,都属于线程私有,生命周期与线程相同,描述的是java方法执行的内存模型,每个方法执行都会创建一个栈帧,用于存储局部变量表,操作栈,动态链接,方法出口等信息,每一个方法被调用直至执行完成的过程,就对应一个栈帧在虚拟机栈从入栈到出栈的过程。局部变量表存放了编译期可知的各种数据基本类型(Boolean,byte,char,short,int,float,long,double),以及对象的引用。

  这个区域规定了两种异常状况:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;虚拟机栈在进行动态扩展时,无法申请到足够的内存,将抛出OutOfMemoryError异常。

  2.3、本地方法栈(Native Method Stack)

  本地方法栈与虚拟机栈所发挥的作用非常相似,他们之间区别不过是虚拟机栈为虚拟机执行Java方法(也就是字节码)服务,而本地方法栈则为虚拟机使用到的Native方法服务。

  举例(StackOverflowError):

  

 1 public class TestStackSOF {
 2     public long stackLenth = 1;
 3
 4     public void stackSOF(){
 5         stackLenth++;
 6         System.out.println(stackLenth);
 7         stackSOF();
 8     }
 9     public static void main(String[] args) {
10         TestStackSOF tss = new TestStackSOF();
11         try{
12             tss.stackSOF();
13         }catch(Throwable e){
14             System.out.println("stackLenth: "+tss.stackLenth);
15             try {
16                 throw e;
17             } catch (Throwable e1) {
18                 // TODO Auto-generated catch block
19                 e1.printStackTrace();
20             }
21         }
22     }
23 }

  执行的结果:

  stackLenth: 326323

  java.lang.StackOverflowError

  at com.cn.TestStackSOF.stackSOF(TestStackSOF.java:9)

  at com.cn.TestStackSOF.stackSOF(TestStackSOF.java:9)

  如果把-Xss调到50M,执行的结果是:

  stackLenth: 1637043

  java.lang.StackOverflowError

  at com.cn.TestStackSOF.stackSOF(TestStackSOF.java:9)

   at com.cn.TestStackSOF.stackSOF(TestStackSOF.java:9)

  总结:不难看出,-Xss大小不一样,执行的结果也不一样。如果以后在项目中遇到java.lang.StackOverflowError异常,可以先检查代码是否有无限递归,如果不是,可加大-Xss大小再看运行效果。

  2.4、Java堆(Heap)

  java堆是Java虚拟机所管理的内存中最大的一块,被所有线程共享的内存区域。此区域唯一的目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。

  java堆是垃圾收集器管理的主要区域,也叫做”GC堆“。

  java堆的大小可扩展,通过-Xmx和-Xms控制,如果在堆中没有内存完成实例分配,并且堆也无法再扩展时,将会抛出OutOfMemoryError异常。

  举例(OutOfMemoryError):

  

import java.util.ArrayList;
import java.util.List;

public class TestHeapOOM {
    /**
     * vm args -Xmn120M -Xmx1024M
     */
    public static void main(String[] args) {
        List<String> list = new ArrayList<String>();
        while(true){
            list.add("sss");
        }
    }
}

  运行结果:

  Exception in thread "main" java.lang.OutOfMemoryError: Java heap space

  at java.util.Arrays.copyOf(Unknown Source)

  2.5、方法区(Method Area)

  方法区与Java堆一样,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。方法区也叫做”永久代“,并非是指数据永久存在。该区域内存回收目标主要是针对常量池的回收和对类型的卸载。这个区域的回收成绩比较难以令人满意,尤其是类型的卸载,条件相当苛刻。但这部分内存回收是必要存在的。当方法区无法满足内存分配需求时,将抛出OutOfMemoryError异常。

  2.6、直接内存(Direct Area)

  直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。但是这部分被频繁的使用,也可能导致OutOfMemoryError异常出现。

  直接内存不受Java堆大小的限制,在JDK1.4中新加入了NIO(New Input/Output)类,NIO的Buffer提供一个可以直接访问系统物理内存的类——DirectBuffer。DirectBuffer类继承自ByteBuffer,但和普通的ByteBuffer不同。普通的ByteBuffer仍在JVM堆上分配内存,其最大内存受到最大堆内存的限制。而DirectBuffer直接分配在物理内存中,并不占用堆空间。在访问普通的ByteBuffer时,系统总是会使用一个“内核缓冲区”进行操作。而DirectBuffer所处的位置,就相当于这个“内核缓冲区”。因此,使用DirectBuffer是一种更加接近内存底层的方法,所以它的速度比普通的ByteBuffer更快。

时间: 2024-08-01 10:30:47

《深入理解java虚拟机》笔记(1)运行时数据区域的相关文章

Java虚拟机学习--记录运行时数据区域

为方便后面学习的理解,记录一下! 运行时数据区 1.线程共享 1.1方法区(Method Area) 1.1.1运行时常量池(Runtime Constant Pool) 1.2堆(Heap) 2.线程私有 2.1虚拟机栈(VM Stack) 2.2本地方法栈(Native Method Stack) 2.3程序计数器(Program Counter Register) 3.直接内存(Direct Memory) 虚拟机栈: 线程私有,生命周期与线程同步,用来执行Java方法. 每个java方法

你还在看《深入理解Java虚拟机》的运行时数据模型吗?

学习JVM必看的书籍无疑是<深入理解Java虚拟机>这本书了,在书中,关于运行时数据区域模型是这样描述的: 在这里我们只针对HotSpot VM来说,它是OracleJDK和OpenJDK中所带的虚拟机,也是目前使用范围最广的Java虚拟机.在JDK7之前,这样的模型是正确的.但是到了JDK8,如图标红的部分,做了一些优化. 什么是方法区,什么是永久代,运行时常量池又是什么 "方法区"(Method Area),是线程共享的区域,用于存储已被虚拟机加载的类信息,常量,静态变

深入理解Java虚拟机读书笔记---运行时数据区域

运行时数据区域 1.程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器.字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,分支.循环.跳转.异常处理.线程恢复等基础功能都需要依赖这个计数器来完成.由于Java虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器(对于多核处理器来说是一个内核)都只会执行一条线程中的指令.因此,为了线

深入理解Java虚拟机:运行时数据区域

Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有些区域则依赖用户线程的启动和结束而建立和销毁.根据<Java虚拟机规范(Java SE 7版)>的规定,Java虚拟机所管理的内存将会包括以下几个运行时数据区域. 程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器.在虚拟机的概念模型里

深入理解java虚拟机笔记(一)-java内存区域与内存溢出

1. 前言 这是深入理解java虚拟机一书的笔记,来自第二章.因为这本书讲的比较深奥,这是第二次看,需要记录一下笔记. 2. 运行时数据区域 java虚拟机所管理的内存分为以下几个区域. ps:图片来自网络 2.1 程序计数器 程序计数器是一块较小的内存空间,他可以看做是当前线程所执行字节码的行号指示器.字节码解释器工作时就是通过改变这个计数器的值来选去下一条要执行的字节码指令,分之.循环.跳转.异常处理.线程恢复等基础功能都需要依赖这个计数器来完成. 这块内存是线程私有的内存. 如果线程在执行

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

JVM执行Java程序的过程中,会使用到各种数据区域,这些区域有各自的用途.创建和销毁时间.根据<Java虚拟机规范>,JVM包括下列几个运行时数据区域,如下图所示: 其中红色部分是线程私有的,即每个线程各自都有自己的一份.绿色部分是各个线程共享的. 1.PC寄存器(The pc Register) (1)每一个Java线程都有一个PC寄存器. (2)PC寄存器是用于存储每个线程下一步将执行的JVM指令,如该方法为native的,则PC寄存器中不存储任何信息. (3)此内存区域是唯一一个在JV

Java虚拟机 - 结构原理与运行时数据区域

http://liuwangshu.cn/java/jvm/1-runtime-data-area.html 前言 本来计划要写Android内存优化的,觉得有必要在此之前介绍一下Java虚拟机的相关知识,Java虚拟机也并不是三言两语能够介绍完的,因此开了Java虚拟机系列,这一篇文章我们来学习Java虚拟机的结构原理与运行时数据区域. 1.Java虚拟机概述 Oracle官方定义的Java技术体系主要包括以下几个部分: Java程序设计语言 各种平台的Java虚拟机 Class文件格式 Ja

Java虚拟机(一)结构原理与运行时数据区域

前言 本来计划要写Android内存优化的,觉得有必要在此之前介绍一下Java虚拟机的相关知识,Java虚拟机也并不是三言两语能够介绍完的,因此开了Java虚拟机系列,这一篇文章我们来学习Java虚拟机的结构原理与运行时数据区域. 1.Java虚拟机概述 Oracle官方定义的Java技术体系主要包括以下几个部分: Java程序设计语言 各种平台的Java虚拟机 Class文件格式 Java API类库 第三方Java类库 可以把Java程序设计语言.Java虚拟机和Java API类库这三部分

Java虚拟机运行时数据区域

Java虚拟机在执行Java程序的过程中会将其管理的内存划分为若干个不同的数据区域,这些区域有各自的用途,及创建和销毁的时间,有些区域随虚拟机进程的启动而存在,有些区域则是依赖用户线程的启动和结束来建立和销毁.Java虚拟机所管理的内存包括以下几个运行时数据区域,如图(图片引自网络): 1.1 程序计数器(Program Counter Register) 程序计数器是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码的信号指示器.字节码解释器就是通过改变该计数器的值来选取下一条需要执