java内存区域及对象

标签: jvm
10人阅读 评论(0) 收藏 举报
分类:

一.运行时数据区域

java虚拟机(jvm)定义了程序在运行时会用到的内存区域,如下图所示:

这里写图片描述

按照线程间是否有共享,可分为如下几类:
1. 线程独有的内存区域
    1)JAVA STACK,虚拟机栈:用于存储局部变量表、操作数栈、动态链接、方法出口等信息,每一个方法从调用直至执行完毕的过程,就对应着一个栈帧在虚拟机中入栈到出栈的过程。栈的大小和具体JVM的实现有关,通常在256K~756K之间。
    2)NATIVE METHOD STACK,方法栈:和虚拟机栈起的作用一样,只不过方法栈为虚拟机使用到的Native方法服务。虚拟机规范并没有对这个区域有什么强制规定,因此我们使用的HotSpot虚拟机,就干脆没有这块区域了,它和虚拟机栈是一起的。
    3)PROGRAM COUNTER REGISTER,程序计数器:这块内存区域很小,它是当前线程所执行的字节码的行号指示器,字节码解释器通过改变这个计数器的值来选取下一条需要执行的字节码指令。Java方法这个计数器才有值,如果执行的是一个Native方法,那这个计数器是空的。
2. 线程间共享的内存区域
    1)METHOD AREA,方法区:这块区域用于存储虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据,虚拟机规范是把这块区域描述为堆的一个逻辑部分的,但实际它应该是要和堆区分开的。从上面提到的分代收集算法的角度看,HotSpot中,方法区≈永久代。不过JDK 7之后,我们使用的HotSpot应该就没有永久代这个概念了,会采用Native Memory来实现方法区的规划了。
    2)RUNTIME CONSTANT POOL,运行时常量池:上面的图中没有画出来,因为它是方法区的一部分。Class文件中除了有类的版本信息、字段、方法、接口等描述信息外,还有一项信息就是常量池,用于存放编译期间生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运行时常量池中,另外翻译出来的直接引用也会存储在这个区域中。这个区域另外一个特点就是动态性,Java并不要求常量就一定要在编译期间才能产生,运行期间也可以在这个区域放入新的内容,String.intern()方法就是这个特性的应用。
    3)HEAP,堆:大多数应用,堆都是Java虚拟机所管理的内存中最大的一块,它在虚拟机启动时创建,此内存唯一的目的就是存放对象实例。由于现在垃圾收集器采用的基本都是分代收集算法,所以堆还可以细分为新生代和老年代,再细致一点还有Eden区、From Survivior区、To Survivor区,这个后面都会讲到的。
3. 直接内存
    直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。但是这部分内存也被频繁地使用,而且也可能导致内存溢出问题。JDK1.4中新增加了NIO,引入了一种基于通道与缓冲区的I/O方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。这样能在一些场景中显著提高性能,因为避免了在Java堆和Native堆中来回复制数据。显然,本机直接内存的分配不会受到Java堆大小的限制,但是,既然是内存,肯定还是会受到本机总内存(包括RAM、SWAP区)大小以及处理器寻址空间的限制。

二.对象创建

    创建对象在代码中就是通过一个new关键字来实现,而在虚拟机中就没有这么简单了,分为下面几步:
1. 虚拟机遇到new指令后,会先解析new指令后面所带的参数,通过这些参数看能否在常量池中定位到一个类的符号引用,并检查这个符号引用代表的类是否已经加载,解析,初始化,如果没有,会先执行类的初始化过程。
2. 类加载检查通过后,虚拟机就会为新生对象分配内存。为对象分配的内存大小在类加载完成后就确定了,此时就是直接从堆中分出一块确定大小的内存而已,这时会有2个问题:
    1)如果内存是规整的,那么虚拟机将采用的是指针碰撞法来为对象分配内存。意思是所有用过的内存在一边,空闲的内存在另外一边,中间放着一个指针作为分界点的指示器,分配内存就仅仅是把指针向空闲那边挪动一段与对象大小相等的距离罢了。如果垃圾收集器选择的是Serial、ParNew这种基于压缩算法的,虚拟机采用这种分配方式。
    2)如果内存不是规整的,已使用的内存和未使用的内存相互交错,那么虚拟机将采用的是空闲列表法来为对象分配内存。意思是虚拟机维护了一个列表,记录上哪些内存块是可用的,再分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的内容。如果垃圾收集器选择的是CMS这种基于标记-清除算法的,虚拟机采用这种分配方式。
    另外一个问题就是保证new对象时候的线程安全性。因为可能出现虚拟机正在给对象A分配内存,指针还没有来得及修改,对象B又同时使用了原来的指针来分配内存的情况。虚拟机采用了CAS配上失败重试的方式保证更新更新操作的原子性和TLAB两种方式来解决这个问题。
3. 内存分配结束,虚拟机将分配到的内存空间都初始化为零值(不包括对象头)。这一步保证了对象的实例字段在Java代码中可以不用赋初始值就可以直接使用,程序能访问到这些字段的数据类型所对应的零值。
4. 对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的GC分代年龄等信息,这些信息存放在对象的对象头中。
5. 执行<init>方法,把对象按照程序员的意愿进行初始化,这样一个真正可用的对象才算完全产生出来。
    以上这部分内容,如果有下载OpenJDK8的源代码的话,可以通过参考hotspot/src/share/vm/interpreter/bytecodeInterpreter.cpp文件,从2161行开始。2161行的代码是CASE(_new):{…},意思是当代码中遇见new这个关键字,虚拟机做的事情。实际虚拟机可能并不是执行的这段代码,但是通过这段代码来了解new对象的时候虚拟机的运作过程基本上是没问题的。
    
这里附上下载OpenJDK8的链接    密码:6iiw

三.对象的定位方式

建立对象是为了使用对象,Java程序需要通过栈上的reference(引用)数据来操作堆上的具体对象。比如我们写了一句
Object obj = new Object()
而new Object()之后其实有两部分内容,一部分是类数据(比如代表类的Class对象)、一部分是实例数据,由于reference在Java虚拟机规范中只是一个指向对象new Object()的引用obj,并没有规定obj应该通过何种方式去定位、访问堆中对象的具体位置,所以对象访问方式也是取决于虚拟机而定的。主流方式有两种:
1、句柄访问。Java堆中划分出一块句柄池,obj指向的是对象的句柄地址,句柄中则包含了类数据的地址和实例数据的地址。
2、指针访问。对象中存储所有的实例数据和类数据的地址,obj指向的是这个对象。
HotSpot虚拟机采用的是后者,不过前者的对象访问方式也是十分常见的。

查看评论

Java内存对象及区域

几个计算机的概念 为以后写文章考虑,也为巩固自己的知识和一些基本概念,这里要理清楚几个计算机中的概念。 1、计算机存储单位 从小到大依次为位Bit、字节Byte、千字节KB、兆M、千兆GB、TB...
  • zhangyuan19880606
  • zhangyuan19880606
  • 2016-04-20 18:19:20
  • 650

JVM线程内存区域划分

JVM内存区域划分Eden Space、Survivor Space、Tenured Gen,Perm Gen解释
  • zly9923218
  • zly9923218
  • 2016-03-21 14:57:21
  • 608

Java内存区域简述

在Java中,内存由虚拟机进行管理,不像C++那样需要为每个对象写相应的delete/free方法,不容易出现内存泄漏和内存溢出的问题。 Java虚拟机在执行java程序的过程中会把他所管理的内存划...
  • A199581
  • A199581
  • 2017-08-12 15:17:53
  • 137

Java JVM虚拟机7块内存区划分【入门】

转载出处:http://blog.csdn.net/junlinbo/article/details/17712235 顺口溜:器池堆,栈栈区区                 弃池堆,站...
  • wonad12
  • wonad12
  • 2018-01-03 14:25:46
  • 232

Java内存区域模型、对象创建过程、常见OOM

本文内容来源于《深入理解Java虚拟机》一书,非常推荐大家去看一下这本书。最近开始看这本书,打算再开一个相关系列,来总结一下这本书中的重要知识点。呃呃呃,说好的那个图片请求框架呢~  不要急哈,因...
  • u014082714
  • u014082714
  • 2016-06-27 21:31:58
  • 1520

Java内存区域理解

Java内存区域总览: Java虚拟机包括下面几个运行时数据区域:   Java虚拟机在执行Java的过程中会把管理的内存划分为若干个不同的数据区域。这些区域有各自的用途,...
  • hanyuliang2000
  • hanyuliang2000
  • 2016-07-04 09:18:59
  • 407

java虚拟机学习之线程共享内存区和线程私有区

线程共享指的就是可以允许被所有线程共享访问的一块内存,包括堆区,方法区和运行时常量池。  1. java堆区      java堆区在虚拟机启动时被创建,并且他在实际内存中是可以不连续的。java...
  • likailonghaha
  • likailonghaha
  • 2016-11-07 13:55:50
  • 3003

深入理解java虚拟机系列(一):java内存区域与内存溢出异常

主要是阅读《深入理解java虚拟机:JVM高级特性与最佳实践》第二章:Java内存区域与内存溢出异常的笔记。...
  • l271640625
  • l271640625
  • 2014-10-03 23:48:45
  • 2253

chap9 虚拟内存

chap9 虚拟内存 virtual memory CSAPP 读书笔记系列chap9
  • ferris_chan
  • ferris_chan
  • 2017-12-22 16:42:45
  • 93

JAVA内存管理【1.3 JVM的内存区域组成】

1.3  JVM的内存区域组成 java把内存分两种:一种是栈内存,另一种是堆内存         1。在函数中定义的基本类型变量和对象的引用变量都在函数的栈内存中分配;         2。堆内存用...
  • mexican_ok
  • mexican_ok
  • 2013-11-29 16:31:54
  • 692
    个人资料
    等级:
    访问量: 12万+
    积分: 1577
    排名: 3万+
    最新评论