`
SIHAIloveYAN
  • 浏览: 119322 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类

深入理解Java虚拟机-Java内存区域透彻分析

    博客分类:
  • java
阅读更多

Java虚拟机深入理解系列全部文章更新中…

这篇文章主要介绍Java内存区域,也是作为Java虚拟机的一些最基本的知识,理解了这些知识之后,才能更好的进行Jvm调优或者更加深入的学习,本来这些知识是晦涩难懂的,所以希望能够讲解的透彻且形象。

0 运行时数据区域

JVM载执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。

Java 虚拟机所管理的内存一共分为Method Area(方法区)、VM Stack(虚拟机栈)、Native Method Stack(本地方法栈)、Heap(堆)、Program Counter Register(程序计数器)五个区域。

这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有些区域则是依赖用户线程的启动和结束而建立和销毁。具体如下图所示:

上图介绍的是JDK1.8 JVM运行时内存数据区域划分。1.8同1.7比,最大的差别就是:元数据区取代了永久代。元空间的本质和永久代类似,都是对JVM规范中方法区的实现。不过元空间与永久代之间最大的区别在于:元数据空间并不在虚拟机中,而是使用本地内存

1 程序计数器(Program Counter Register)

程序计数器(Program Counter Register)是一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器。在虚拟机概念模型中,字节码解释器工作时就是通过改变计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。

程序计数器是一块 “线程私有” 的内存,每条线程都有一个独立的程序计数器,能够将切换后的线程恢复到正确的执行位置。

  • 执行的是一个Java方法

计数器记录的是正在执行的虚拟机字节码指令的地址

  • 执行的是Native方法

计数器为空(Undefined),因为native方法是java通过JNI直接调用本地C/C++库,可以近似的认为native方法相当于C/C++暴露给java的一个接口,java通过调用这个接口从而调用到C/C++方法。由于该方法是通过C/C++而不是java进行实现。那么自然无法产生相应的字节码,并且C/C++执行时的内存分配是由自己语言决定的,而不是由JVM决定的。

  • 程序计数器也是唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的内存区域。

其实,我感觉这块区域,作为我们开发人员来说是不能过多的干预的,我们只需要了解有这个区域的存在就可以,并且也没有虚拟机相应的参数可以进行设置及控制。

2 Java虚拟机栈(Java Virtual Machine Stacks)

Java虚拟机栈(Java Virtual Machine Stacks)描述的是Java方法执行的内存模型:每个方法在执行的同时都会创建一个栈帧(Stack Frame),从上图中可以看出,栈帧中存储着局部变量表操作数栈动态链接方法出口等信息。每一个方法从调用直至执行完成的过程,会对应一个栈帧在虚拟机栈中入栈到出栈的过程。

与程序计数器一样,Java虚拟机栈也是线程私有的。

局部变量表中存放了编译期可知的各种:

  • 基本数据类型(boolen、byte、char、short、int、 float、 long、double)
  • 对象引用(reference类型,它不等于对象本身,可能是一个指向对象起始地址的指针,也可能是指向一个代表对象的句柄或其他与此对象相关的位置)
  • returnAddress类型(指向了一条字节码指令的地址)

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

Java虚拟机规范中对这个区域规定了两种异常状况:

  • StackOverflowError:线程请求的栈深度大于虚拟机所允许的深度,将会抛出此异常。
  • OutOfMemoryError:当可动态扩展的虚拟机栈在扩展时无法申请到足够的内存,就会抛出该异常。

一直觉得上面的概念性的知识还是比较抽象的,下面我们通过JVM参数的方式来控制栈的内存容量,模拟StackOverflowError异常现象。

3 本地方法栈(Native Method Stack)

本地方法栈(Native Method Stack) 与Java虚拟机栈作用很相似,它们的区别在于虚拟机栈为虚拟机执行Java方法(即字节码)服务,而本地方法栈则为虚拟机使用到的Native方法服务。

在虚拟机规范中对本地方法栈中使用的语言、方式和数据结构并无强制规定,因此具体的虚拟机可实现它。甚至有的虚拟机(Sun HotSpot虚拟机)直接把本地方法栈和虚拟机栈合二为一。与虚拟机一样,本地方法栈会抛出StackOverflowErrorOutOfMemoryError异常。

这个例子中,我们将栈内存的容量设置为256K(默认1M),并且再定义一个变量查看栈递归的深度。

 1/**
 2 * @ClassName Test_02
 3 * @Description 设置Jvm参数:-Xss256k
 4 * @Author 欧阳思海
 5 * @Date 2019/9/30 11:05
 6 * @Version 1.0
 7 **/
 8public class Test_02 {
 9
10    private int len = 1;
11
12    public void stackTest() {
13        len++;
14        System.out.println("stack len:" + len);
15        stackTest();
16    }
17
18    public static void main(String[] args) {
19        Test_02 test = new Test_02();
20        try {
21            test.stackTest();
22        } catch (Throwable e) {
23            e.printStackTrace();
24        }
25    }
26}

运行时设置JVM参数

输出结果:

4 Java堆(Heap)

对于大多数应用而言,Java堆(Heap)是Java虚拟机所管理的内存中最大的一块,它被所有线程共享的,在虚拟机启动时创建。此内存区域唯一的目的存放对象实例,几乎所有的对象实例都在这里分配内存,且每次分配的空间是不定长的。在Heap 中分配一定的内存来保存对象实例,实际上只是保存对象实例的属性值属性的类型对象本身的类型标记等,并不保存对象的方法(方法是指令,保存在Stack中),在Heap 中分配一定的内存保存对象实例和对象的序列化比较类似。

Java堆是垃圾收集器管理的主要区域,因此也被称为 “GC堆(Garbage Collected Heap)” 。从内存回收的角度看内存空间可如下划分:

图片摘自https://blog.csdn.net/bruce128/article/details/79357870图片摘自https://blog.csdn.net/bruce128/article/details/79357870

  • 新生代(Young): 新生成的对象优先存放在新生代中,新生代对象朝生夕死,存活率很低。在新生代中,常规应用进行一次垃圾收集一般可以回收70% ~ 95% 的空间,回收效率很高。

如果把新生代再分的细致一点,新生代又可细分为Eden空间From Survivor空间To Survivor空间,默认比例为8:1:1。

  • 老年代(Tenured/Old):在新生代中经历了多次(具体看虚拟机配置的阀值)GC后仍然存活下来的对象会进入老年代中。老年代中的对象生命周期较长,存活率比较高,在老年代中进行GC的频率相对而言较低,而且回收的速度也比较慢。
  • 永久代(Perm):永久代存储类信息、常量、静态变量、即时编译器编译后的代码等数据,对这一区域而言,Java虚拟机规范指出可以不进行垃圾收集,一般而言不会进行垃圾回收。

其中新生代和老年代组成了Java堆的全部内存区域,而永久代不属于堆空间,它在JDK 1.8以前被Sun HotSpot虚拟机用作方法区的实现

另外,再强调一下堆空间内存分配的大体情况,这对于后面一些Jvm优化的技巧还是有帮助的。

  • 老年代 : 三分之二的堆空间
  • 年轻代 : 三分之一的堆空间
    eden区: 8/10 的年轻代空间
    survivor0 : 1/10 的年轻代空间
    survivor1 : 1/10 的年轻代空间

最后,我们再通过一个简单的例子更加形象化的展示一下堆溢出的情况。

  • JVM参数设置:-Xms10m -Xmx10m

这里将堆的最小值和最大值都设置为10m,如果不了解这些参数的含义,可以参考这篇文章:深入理解Java虚拟机-常用vm参数分析

 1/**
 2 * VM Args:-Xms10m -Xmx10m -XX:+HeapDumpOnOutOfMemoryError
 3 * @author zzm
 4 */
 5public class HeapTest {
 6
 7    static class HeapObject {
 8    }
 9
10    public static void main(String[] args) {
11        List<HeapObject> list = new ArrayList<HeapObject>();
12
13        //不断的向堆中添加对象
14        while (true) {
15            list.add(new HeapObject());
16        }
17    }
18}

输出结果:

图中出现了java.lang.OutOfMemoryError,并且提示了Java heap space,这就说明是Java堆内存溢出的情况。

堆的Dump文件分析

我的使用的是VisualVM工具进行分析,关于如何使用这个工具查看这篇文章(深入理解Java虚拟机-如何利用VisualVM对高并发项目进行性能分析 )。在运行程序之后,会同时打开VisualVM工具,查看堆内存的变化情况。

在上图中,可以看到,堆的最大值是30m,但是使用的堆的容量也快接近30m了,所以很容易发生堆内存溢出的情况。

接着查看dump文件。

如上图,堆中的大部分的对象都是HeapObject,所以,就是因为这个对象的一直产生,所以导致堆内存不够分配,所以出现内存溢出。

我们再看GC情况。

如上图,Eden新生代总共48次minor gc,耗时1.168s,基本满足要求,但是survivor却没有,这不正常,同时Old Gen老年代总共27次full gc,耗时4.266s,耗时长,gc多,这正是因为大量的大对象进入到老年代导致的,所以,导致full gc频繁。

5 方法区(Method Area)

方法区(Method Area) 与Java堆一样,是各个线程共享的内存区域。它用于存储一杯虚拟机加载类信息、常量、静态变量、及时编译器编译后的代码等数据。正因为方法区所存储的数据与堆有一种类比关系,所以它还被称为 Non-Heap

运行时常量池(Runtime Constant Pool)

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

Java虚拟机对Class文件每一部分(自然包括常量池)的格式有严格规定,每一个字节用于存储那种数据都必须符合规范上的要求才会被虚拟机认可、装载和执行。但对于运行时常量池,Java虚拟机规范没有做任何有关细节的要求,不同的提供商实现的虚拟机可以按照自己的需求来实现此内存区域。不过一般而言,除了保存Class文件中的描述符号引用外,还会把翻译出的直接引用也存储在运行时常量池中。

运行时常量池相对于Class文件常量池的另外一个重要特征是具备动态性,Java语言并不要求常量一定只有编译器才能产生,也就是并非置入Class文件中的常量池的内容才能进入方法区运行时常量池,运行期间也可能将新的常量放入池中

运行时常量池举例

上面的动态性在开发中用的比较多的便是String类的intern() 方法。所以,我们以intern() 方法举例,讲解一下运行时常量池

String.intern()是一个native方法,作用是:如果字符串常量池中已经包含有一个等于此String对象的字符串,则直接返回池中的字符串;否则,加入到池中,并返回。

 1/**
 2 * @ClassName MethodTest
 3 * @Description vm参数设置:-Xms512m -Xmx512m -Xmn128m -XX:PermSize=10M -XX:MaxPermSize=10M -XX:NewRatio=4 -XX:SurvivorRatio=8 -XX:MaxTenuringThreshold=15 -XX:-HeapDumpOnOutOfMemoryError -XX:+UseParNewGC -XX:+UseConcMarkSweepGC
 4 * @Author 欧阳思海
 5 * @Date 2019/11/25 20:06
 6 * @Version 1.0
 7 **/
 8
 9public class MethodTest {
10
11    public static void main(String[] args) {
12        List<String> list = new ArrayList<String>();
13        long i = 0;
14        while (i < 1000000000) {
15            System.out.println(i);
16            list.add(String.valueOf(i++).intern());
17        }
18    }
19}

vm参数介绍:

-Xms512m -Xmx512m -Xmn128m -XX:PermSize=10M -XX:MaxPermSize=10M -XX:NewRatio=4 -XX:SurvivorRatio=8 -XX:MaxTenuringThreshold=15 -XX:-HeapDumpOnOutOfMemoryError -XX:+UseParNewGC -XX:+UseConcMarkSweepGC
开始堆内存和最大堆内存都是512m,永久代大小10m,新生代和老年代1:4,E:S1:S2=8:1:1,最大经过15次survivor进入老年代,使用的,垃圾收集器是新生代ParNew,老年代CMS。

通过这样的设置之后,查看运行结果:

首先堆内存耗完,然后看看GC情况,设置这些参数之后,GC情况应该会不错,拭目以待。

上图是GC情况,我们可以看到新生代 21 次minor gc,用了1.179秒,平均不到50ms一次,性能不错,老年代 117 次full gc,用了45.308s,平均一次不到1s,性能也不错,说明jvm运行是不错的。

注意: 在JDK1.6及以前的版本中运行以上代码,因为我们通过-XX:PermSize=10M -XX:MaxPermSize=10M设置了方法区的大小,所以也就是设置了常量池的容量,所以运行之后,会报错:java.lang.OutOfMemoryError:PermGen space,这说明常量池溢出;在JDK1.7及以后的版本中,将会一直运行下去,不会报错,在前面也说到,JDK1.7及以后,去掉了永久代。

6 直接内存

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

这个我们实际中主要接触到的就是NIO,在NIO中,我们为了能够加快IO操作,采用了一种直接内存的方式,使得相比于传统的IO快了很多。

在NIO引入了一种基于通道(Channel)与缓冲区(Buffer)的I/O方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。这样能避免在Java堆和Native堆中来回复制数据,在一些场景里显著提高性能。

在配置虚拟机参数时,会根据实际内存设置-Xmx等参数信息,但经常忽略直接内存,使得各个内存区域总和大于物理内存限制(包括物理的和操作系统的限制),从而导致动态扩展时出现OutOfMemoryError异常。

1、原创不易,老铁,文章需要你的点赞让更多的人看到,希望能够帮助到大家!

2、文章有不当之处,欢迎指正,如果喜欢微信阅读,你也可以关注我的微信公众号好好学java,公众号已有 6W 粉丝,回复:1024,获取公众号的大礼包,公众号长期发布 Java 优质系列文章,关注我们一定会让你收获很多!

0
1
分享到:
评论

相关推荐

    深入理解Java虚拟机-Java内存区域透彻分析(序列化、反序列化概念及其使用场景+实现序列化的方式+transient关键字)

    深入理解Java虚拟机-Java内存区域透彻分析(序列化、反序列化概念及其使用场景+实现序列化的方式+transient关键字) Java序列化和反序列化是Java虚拟机中的一种重要机制,它们可以将Java对象转换为二进制数据,然后...

    深入Java虚拟机(原书第2版)

    《深入理解Java虚拟机:JVM高级特性与最佳实践》内容简介:作为一位Java程序员,你是否也曾经想深入理解Java虚拟机,但是却被它的复杂和深奥拒之门外?没关系,本书极尽化繁为简之妙,能带领你在轻松中领略Java虚拟机...

    java虚拟机源码学习-UnderstandingTheJVM:深入理解Java虚拟机(周志明)源码及学习笔记

    《深入理解Java虚拟机》是Java开发者们深入探讨Java运行机制的经典之作,作者周志明以其深入浅出的讲解方式,揭示了Java虚拟机(JVM)的工作原理。本资源包含该书第三版的源码分析及学习笔记,旨在帮助读者更透彻地...

    深入了解java虚拟机

    本书是近年来国内出版的唯一一本与Java虚拟机相关的专著,也是唯一一本同时从核心理论和实际运用这两个角度去探讨Java虚拟机的著作,不仅理论分析得透彻,而且书中包含的典型案例和最佳实践也极具现实指导意义

    深入理解Android:卷I--详细书签版

     《深入理解android:卷1》是一本以情景方式对android的源代码进行深入分析的书。内容广泛,以对framework层的分析为主,兼顾native层和application层;分析深入,每一部分源代 码的分析都力求透彻;针对性强,...

    Java-Interview-Question.pdf

    在Java面试中,通常会涉及一系列的基础知识点和概念,以考察应聘者是否具备Java编程的扎实基础和对面向对象编程的深入理解。以下是根据提供的文件内容整理的Java知识点详解: ### Java语言特点 Java是一种面向对象...

    深入理解Android 卷1.pdf

    分析深入,每一部分源代码的分析都力求透彻;针对性强,注重实际应用开发需求,书中所涵盖的知识点都是Android应用开发者和系统开发者需要重点掌握的。共10章,第1章介绍了阅读本书所需要做的准备工作,主要包括对...

    ali-java-interview.pdf

    在探讨Java面试相关知识点时...在准备Java面试时,不仅要对上述知识点有一个全面的了解,而且需要深入理解其原理和使用场景。这样才能在面试中对各种问题给出深入透彻的回答,展示出你深厚的技术底蕴和解决问题的能力。

    深入理解Android:卷2

    深入理解Android:卷2》是一本以情景方式对Android的源代码进行深入分析的书。内容广泛,以对Framework层的分析为主,兼顾Native层和Application层;分析深入,每一部分源代码的分析都力求透彻;针对性强,注重实际...

    深入理解Android卷1

    深入理解Android:卷I》是一本以情景方式对Android的源代码进行深入分析的书。内容广泛,以对Framework层的分析为主,兼顾Native层和Application层;分析深入,每一部分源代码的分析都力求透彻;针对性强,注重实际...

    java小教程

    深入虚拟机的小教程,理解更透彻!希望读者能好好的深入java虚拟机的知识,尽快的把java理解的更透彻!

    疯狂的Java

    最后,书中还包含了JVM(Java虚拟机)的相关知识,包括内存管理、垃圾收集机制、类加载器等,这些都是优化Java程序性能和排查问题的关键。 总的来说,《疯狂的Java》全面覆盖了Java编程的各个方面,无论是对新手...

    java教师上课用讲稿

    10. **Java虚拟机(JVM)**:深入讲解JVM的工作原理,包括类加载、内存模型、垃圾收集等。 11. **Java泛型**:泛型引入后,提高了代码的类型安全性和重用性,减少类型转换的麻烦。 12. **JDBC与数据库操作**:介绍...

    疯狂的java讲义

    9. **JVM内部机制**:了解Java虚拟机的工作原理,包括内存管理(如垃圾回收)、类加载机制等,有助于优化程序性能。 10. **Java EE相关**:虽然标题并未明确提及,但作为全面的Java学习资料,可能会涉及Servlet、...

    开始java编程第二版(英文版)

    - **平台独立性**:Java程序可以运行在任何安装了Java虚拟机(JVM)的操作系统上,这一特性使得Java成为跨平台开发的理想选择。 - **面向对象**:Java完全支持面向对象编程(OOP)的核心概念,如封装、继承和多态。 - **...

    李兴华java8视频及详细笔记

    这一部分会介绍Java的平台独立性、"一次编写,到处运行"的理念,以及Java虚拟机(JVM)的工作原理,帮助初学者建立对Java的全局认识。 接下来,逐步深入到Java语言的基础语法,包括变量、数据类型、运算符、流程...

    java面试题精选二

    10. **JVM**:了解Java虚拟机的工作原理,包括类加载机制、JVM内存模型、类加载器、方法区的元空间以及垃圾回收策略。 11. **Spring框架**:如果面试题涉及到进阶话题,可能会涵盖Spring框架的基本概念,如依赖注入...

    良葛格Java学习笔记html.rar

    此外,可能还会涉及到JVM(Java虚拟机)的工作原理、垃圾回收机制以及如何进行性能优化等相关知识。 【压缩包子文件的文件名称列表】只有一个文件名为"良葛格Java学习笔记html",这通常意味着压缩包内包含一个完整...

    《深入理解Android:卷I》试读本

    《深入理解Android:卷I》是一本以情景方式对Android的源代码进行深入分析的书。内容广泛,以对Framework层的分析为主,兼顾Native层和Application层;分析深入,每一部分源代码的分析都力求透彻;针对性强,注重实际...

Global site tag (gtag.js) - Google Analytics