在内存管理方面,Java相对于C和C++的区别在于Java具有内存动态分配以及垃圾收集技术,但平时我们很少去关注JVM的内存结构以及GC,在出现内存泄露或溢出方面的问题,排查工作将变得异常艰难。

 
        1. 运行时数据区域
 
                Java虚拟机在执行Java程序时会将其管理的内存按照用于划分为若干个不同的数据区域,这些区域有着各自不同的生命周期。根据《JAVA虚拟机规范》,Java虚拟机管理的内存会包含以下几个区域。其中可以分为共享内存区以及线程隔离数据区两个部分。
 
      
                
        2. 程序计数器
 
                程序计数器是一个非常小的内存空间,可以看做当前线程执行的字节码的行号指示器。在虚拟机中,程序的执行,跳转,循环等都需要该计数器的值来选取下一条要执行的指令。程序计数器占用的内存空间非常小,而且通常不会出现OOM或SOF等。
 
                但需要注意的是,程序计数器如果正在执行一个Java方法,则其中保存的是下一条指令的地址;如果执行的是Natvie方法,则为空。
 
        3. JAVA 虚拟机栈
 
                同程序计数器一样,虚拟机栈也是线程私有的,其周期与线程的生命周期相同,其描述的是Java方法执行的内存模型:每个方法执行的时候都会同时创建一个栈帧用于存储局部变量表、操作栈、动态链接、方法出口等信息。每一个方法被调用直到执行完成的过程就对应着一个虚拟机栈从入栈到出栈的过程。
 
                Java内存区通常习惯被分为堆和栈,这种划分方法实际上是比较粗略的,此时所谓的栈就是虚拟机栈。
 
                局部变量表存放了编译期可知的基本数据类型,对象引用和returnAddress类型。
 
                需要注意的是,局部变量表所需的内存空间在编译期内完成分配,当进入一个方法时,这个方法在帧中分配多大的内存空间是完全确定的,在运行时也不会改变。
 
                在JAVA虚拟机规范中,定义了两种异常:如果线程请求深度大于虚拟机允许深度,则会抛出StackOverflowError异常;如果扩展无法申请到足够内存时会抛出OutOfMemoryError异常。
 
                由-Xss设定栈大小
 
// 虚拟机栈和本地方法栈的大小 = 线程允许最大内存 - 最大堆容量 - 最大方法区容量
// 在多线程时,给每个线程分配的栈越大,越容易出现异常
 
        4. 本地方法栈
 
                本地方法栈只为虚拟机使用的Native方法服务,其他类似于虚拟机栈
 
        5. 堆
 
                Java Heap是java内存管理中最大的一部分,Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象的实例。
 
                Java Heap是GC的主要区域,有时候会被成为“GC堆”。Java堆还可以进行细分:新生代和老年代:再细致一点可以划分为Eden空间、From Survivor空间、To Survivor空间等。虽然被划分为多个空间,其存放的仍是对象,目的是为了更好更快的回收内存。
 
                Java堆可以位于不连续的内存空间上,并且可以通过配置进行扩展 -Xmx 和-Xms控制,如果在堆中没有完成内存分配,而且也无法扩展时,会抛出"OutOfMemoryError“异常。
 
                示例:
   
// 堆溢出
// -Xmx 堆最大值
// -Xms 堆最小值
// -XX:+HeapDumpOnOutOfMemoryError 设置虚拟机在异常时转储当前内存堆
// 可以使用Eclipse Memory Analyzer 对转储的堆进行分析
 
        6. 方法区
 
                方法区也是线程共享的内存区域,用于存储加载的类,常量,静态变量和即时编译器编译后的代码。
 
                对于使用HotSpot虚拟机的开发人员来说,其通常将其称为永久代,意为方法区内较少出现垃圾收集,主要是对常量的清理和类的卸载,但回收的内存的效果较其他内存区域要差。
 
        7. 运行时常量池
 
                Runtime constant pool是方法区的一部分,其中会保存在class文件中的字面量和符号引用,常量池在运行时也可以将数据导入
 
        8. 直接内存
 
                直接内存并非虚拟机运行内存的一部分,是属于运行环境的内存区域。
 
                如JAVA引入的NIO,可以直接操作本机内存,虽然不再受JVM堆大小的限制,但仍然受运行环境内存限制,当超限时,也会抛出OOM异常。
 
        9. 对象访问
 
                Object obj = new Object()
 
                Object obj 将会反映在Java栈的本地变量表中,作为一个reference类型数据出现。
 
                new Object 将会反应在堆中,并分配一块堆的结构化内存
 
                java访问对象可能采用直接访问后者句柄访问两种方式,其中直接访问方式速度快,句柄访问稍慢,但句柄中存放稳定的句柄地址。

深入理解Java虚拟机 - 虚拟机内存划分的更多相关文章

  1. 初识:java虚拟机的内存划分

    什么是内存? 内存是计算机中的重要原件,临时存储区域,作用是运行程序.我们编写的程序是存放在硬盘中的,在硬盘中的程序是不会运行的,必须放进内存中才能运行,运行完毕后会清空内存.Java虚拟机要运行程序 ...

  2. java中的内存划分和一个数组的内存图

    内存概述 内存是计算机中的重要原件,临时存储区域,作用是运行程序.我们编写的程序是存放在硬盘中的,在硬盘中的程序是不会运行的,必须放进内存中才能运行,运行完毕后会清空内存   Java虚拟机要运行程序 ...

  3. java虚拟机的内存划分

    为了提高运算效率,就对空间进行不同区域的划分,因为每一片区域都有特定的处理数据方式和内存管理方式. 一. 1.jvm的内存划分: 区域名称 作用 寄存器 给cpu使用,和我们开发无关 本地方法栈 jv ...

  4. Java运行时内存划分

    这篇文章可以说是摘抄自周志明的<深入理解Java虚拟机>,但是加上了自己的理解,印象可以更深些. Java虚拟机在执行Java程序的时候会把他所管理的内存划分为若干个不同的数据区域,各个区 ...

  5. Java运行时内存划分与垃圾回收--以及类加载机制基础

    ----JVM运行时内存划分----不同的区域存储的内容不同,职责因为不同1.方法区:被线程共享,存储被JVM加载的类的信息,常量,静态变量等2.运行时常量池:属于方法区的一部分,存放编译时期产生的字 ...

  6. Java中的内存划分

    Java程序在运行时,需要在内存中分配空间.为了提高运行效率,就对数据进行了不同的空间划分.因为每一片区域都有特定的数据处理方式和内存管理方式. 具体分为5种内存空间: 程序计数器:保证线程切换后能恢 ...

  7. 深入理解 Java 对象的内存布局

    对于 Java 虚拟机,我们都知道其内存区域划分成:堆.方法区.虚拟机栈等区域.但一个对象在 Java 虚拟机中是怎样存储的,相信很少人会比较清楚地了解.Java 对象在 JVM 中的内存布局,是我们 ...

  8. 深入理解Java的堆内存和线程内存

    我们都知道Java对象都是在堆中创建的(开启逃逸分析的情况除外),比如一个线程中有一段这样的代码: public class A{ public int xxx; } 通过A a = new A(); ...

  9. 深入理解Java String#intern() 内存模型

    原文出处: codelog.me 大家知道,Java中string.intern()方法调用会先去字符串常量池中查找相应的字符串,如果字符串不存在,就会在字符串常量池中创建该字符串然后再返回. 字符串 ...

  10. Java虚拟机:内存模型详解

    版权声明:本文为博主原创文章,转载请注明出处,欢迎交流学习! 我们都知道,当虚拟机执行Java代码的时候,首先要把字节码文件加载到内存,那么这些类的信息都存放在内存中的哪个区域呢?当我们创建一个对象实 ...

随机推荐

  1. php异步加载、多线程fsockopen()、fputs()

    index.php <?php function test() { $fp=fsockopen("localhost", 80, $errno, $errstr, 30); ...

  2. Spark小课堂Week6 启动日志详解

    Spark小课堂Week6 启动日志详解 作为分布式系统,Spark程序是非常难以使用传统方法来进行调试的,所以我们主要的武器是日志,今天会对启动日志进行一下详解. 日志详解 今天主要遍历下Strea ...

  3. 关于EndNote X6工具文献管理以及参考文献生成的使用

    1 利用endnote下载参考文献 1.1在CNKI中查找文献,在前面打钩,这里显示已经选中两篇文献了。 然后选择导出文献——选择Endnote——导出并保存text文件 打开Endnote——imp ...

  4. Jar包下载

    到maven上面下载 http://mvnrepository.com/artifact/redis.clients/jedis/2.9.0 到jarfire去下载 http://cn.jarfire ...

  5. DNF技能贴图的研究

    一直在猜想DNF的技能贴图怎么贴的,靠在游戏里慢慢移动确定技能的偏移太费时间了.前段发现了“可视坐标生成”这软件,针对DNF改衣服,装备款式的小工具,就自己写了个类似的. 从图上看,技能的域中心点和人 ...

  6. Netty 4.0 demo

    netty是一个异步,事件驱动的网络编程框架&工具,使用netty,可以快速开发从可维护,高性能的协议服务和客户端应用.是一个继mina之后,一个非常受欢迎的nio网络框架 netty4.x和 ...

  7. 1049: [HAOI2006]数字序列 - BZOJ

    Description 现在我们有一个长度为n的整数序列A.但是它太不好看了,于是我们希望把它变成一个单调严格上升的序列.但是不希望改变过多的数,也不希望改变的幅度太大.Input 第一行包含一个数n ...

  8. PHP session有效期session.gc_maxlifetime的设置方法

    PHP(外文名: Hypertext Preprocessor,中文名:“超文本预处理器”)是一种通用开源脚本语言.语法吸收了C语言.Java和Perl的特点,入门门槛较低,易于学习,使用广泛,主要适 ...

  9. win8.1下使用vmware workstation 来编译ffmpeg

    先吐槽下,T440预装的win8.1 采用 UEFI+GPT分区导致 无法使用传统的EasyBCD来达到win+ubuntu双系统效果, 尝试了2天全部失败,等以后有时间了 买块U盘再干掉win8 现 ...

  10. c++ string assign =

    C++ string类的成员函数,用于拷贝.赋值操作,它们允许我们顺次地把一个string 对象的部分内容拷贝到另一个string 对象上. string &operator=(const s ...