您好,欢迎来到意榕旅游网。
搜索
您的当前位置:首页《深入理解Java虚拟机》读书笔记之Java内存区域

《深入理解Java虚拟机》读书笔记之Java内存区域

来源:意榕旅游网

        在这段笔记上,笔者将从概念上介绍Java虚拟机内存的各个区域,讲解这些区域的作用、服务对象。各个内存区域的实现基于Sun Hotspot虚拟机。


Java运行时数据区

        Java虚拟机在执行Java程序时,会将把其所管理的内存分为若干个数据区域,这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有些区域则依赖用户线程的启动和结束而建立和销毁。根据《Java虚拟机规范(Java SE 7版)》的规定,Java虚拟机所管理的内存将会包括以下几个运行时数据区域:程序计数器、虚拟机栈、Java堆、方法区和本地方法栈,如下图所示:

  • 虚拟机栈(VM Stack):是Java方法执行的内存模型,每个Java方法(或本地方法)在执行的同时会创建栈帧(Stack Frame),用来存放局部变量表、方法出口、操作数栈和动态链接等信息。每个方法从调用到执行完成的过程就对应着这样一个栈帧的入栈和弹栈。局部变量表所需的内存大小在编译期就已分配完成,进入一个方法或方法在执行过程中,其大小是确定的,不变的。Sun Hotspot虚拟机中,Java虚拟机栈和本地方法栈合二为一,此处统称为虚拟机栈。该区域是线程私有的。

  • Java堆(Java Heap):是Java虚拟机所管理内存中最大的一块区域,并且被各线程共享,在虚拟机启动时被创建。此区域唯一目的就是存放对象实例和数组,同时也是GC管理的主要区域。

  • 方法区(Method Area):与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。Java虚拟机规范把方法区描述为堆的一个逻辑部分,亦称“非堆(Non-Heap)”。这区域的内存回收目标主要是针对常量池的回收和对类型的卸载。运行时常量池(Runtime Constant Pool)是方法区的一部分。Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池(Constant Pool Table),用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运行时常量池中存放。

HotSpot虚拟机在Java堆中对象创建(分配内存)、布局和访问

  • 对象的创建:(1)虚拟机遇到一条new指令时,首先将去检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有,那必须先执行相应的类加载过程;(2)类加载检查后,会为对象分配了内存,内存的大小是在类加载完就可以确定的;(3)内存分配完成后,虚拟机需要将分配到的内存空间都初始化为零值(不包括对象头);(4)接下来,虚拟机要对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的GC分代年龄等信息;(5)一般来说,执行new指令之后会接着执行方法,把对象按照程序员的意愿进行初始化,这样一个真正可用的对象才算完全产生出来。

  • 对象的访问定位:Java程序需要通过栈上的reference数据来操作访问堆上的具体对象,目前主流的访问方式有使用句柄和直接指针两种,如图所示

通过句柄访问对象

通过直接指针访问对象

对虚拟机Sun HotSpot而言,它是使用第二种方式进行对象访问的,即通过直接指针访问对象。

因篇幅问题不能全部显示,请点此查看更多更全内容

Copyright © 2019- yrrf.cn 版权所有 赣ICP备2024042794号-2

违法及侵权请联系:TEL:199 1889 7713 E-MAIL:2724546146@qq.com

本站由北京市万商天勤律师事务所王兴未律师提供法律服务