在内存管理方面,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. 《Java并发编程实战》读书笔记(更新中)

    一.简介 1.多线程编程要注意的几点: 安全性:永远不发生糟糕的事情 活跃性:某件正确的事情最终会发生(不会发生无限循环或者死锁) 性能:正确的事尽快发生(上下文切换消耗之类的) 二.线程安全 1.为 ...

  2. zxing-master core编译

    1.下载 Maven: http://maven.apache.org/download.cgi 2.下载完以后,设一下系统变量:MAVE_HOME 3.Path中加入%MAVEN_HOME%\bin ...

  3. MySQL的复制原理及配置

    MySQL 的数据库的高可用性的架构大概有以下几种:集群,读写分离,主备.而后面两种都是通过复制来实现的.下面将简单介绍复制的原理及配置,以及一些常见的问题. 一.复制的原理 MySQL 复制基于主服 ...

  4. 编译升级php

    http://www.linux-centos.com/2014/11/16/%E7%BC%96%E8%AF%91%E5%8D%87%E7%BA%A7%E4%BB%8Ephp5-2-17%E5%88% ...

  5. SVN备份教程(二)

    上次的博文中SVN备份教程(一)我们简单介绍了一下SVN备份是如何操作的,今天我们接着将上次的问题进行优化. 1.问题回顾 在讲之前,我们先来将上次的问题重申一下.之前的SVN备份存在的问题很简单,每 ...

  6. MyBatis对不同数据库的主键生成策略

    本文转自:http://289972458.iteye.com/blog/1001851  http://hi.baidu.com/zim_it/blog/item/8a2bd11205f5b56ec ...

  7. centos7初步配置

    centos7初步配置 首先安装lrzsz zip/unzip yum -y install lrzsz yum -y install zip unzip 安装vim yum install vim* ...

  8. Contest2037 - CSU Monthly 2013 Oct (problem F :ZZY and his little friends)

    http://acm.csu.edu.cn/OnlineJudge/problem.php?cid=2037&pid=5 [题解]: 没想通这题暴力可以过.... [code]: #inclu ...

  9. 使用C#选择文件夹、打开文件夹、选择文件

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...

  10. Nagios3完整配置文档

    第一章:简单快速安装nagios 1.1 准备软件包 在做安装之前确认要对该机器拥有root权限. 确认你安装好的linux系统上已经安装如下软件包再继续. Apache GCC编译器 GD库与开发库 ...