在内存管理方面,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. Python数据结构——二叉树的实现

    1. 二叉树 二叉树(binary tree)中的每个节点都不能有多于两个的儿子. 1.1 二叉树列表实现 如上图的二叉树可用列表表示: tree=['A', #root ['B', #左子树 ['D ...

  2. PHP微信开发代码

    1,SAE上申请服务器 2,绑定测试账号 3,Token验证 <?php /* http://www.cnblogs.com/xrhou12326/ CopyRight 2014 All Rig ...

  3. iOS5编程--ARC在工程上的相关设置

    在创建工程的时候,我们可以指定工程是否使用ARC技术,如下图 选中表示支持ARC, 在Beta5以前的版本中,不提供这个选项,非常麻烦. 如果是你拿到的工程,那么可以通过设置来改变,如下图所示 如果不 ...

  4. Nginx 禁止访问某个目录或文件的设置方法

    如果基于WEB根目录下,要禁止用户访问/config目录,或者要禁止用户访问/config.ini(ZF常用INI,不过建议还是放到WEB目录以外的地方),可以通过location进行配置,返回403 ...

  5. XSS与CSRF两种跨站攻击比较

    XSS:跨站脚本(Cross-site scripting) CSRF:跨站请求伪造(Cross-site request forgery) 在那个年代,大家一般用拼接字符串的方式来构造动态SQL 语 ...

  6. spoj 1437

    自己暴了一下不过     转一个 bfs... #include <cstdio> #include <vector> #include <queue> #incl ...

  7. Topo图

    http://blog.csdn.net/youfangyuan/article/details/8367398 http://joshuaxiao.iteye.com/blog/2224120 ht ...

  8. [itint5]合并K个有序链表

    merge sort,leet code里面曾经做过.但一开始没这么写,遍历来做,效率n*k了,用了merge sort后,变成logn*k. 用了dummy node.同时要注意size为0的情况. ...

  9. nginx负载均衡和反向代理有什么区别

    近在研究nginx的负载均衡和反向代理,先看下这两个简单的配置吧! 负载均衡 worker_processes 1; events { worker_connections 1024; } http{ ...

  10. nginx静态资源分离部署

    修改nginx.conf文件,用于nginx处理静态资源. 主要配置如下(在server配置中加入location配置即可): server { listen 80; server_name 123. ...