一、运行时数据区域

  java虚拟机所管理的内存会包括下面的几个部分:

1.程序计数器:是一块较小的内存空间,可以看做是当前线程所执行的字节码的行号指示器。一般情况下,字节码解释器工作的时候就是通过改变计数器的之来选取需要执行的字节码指令。

  (1)每条线程都有一个独立的程序计数器,每个线程都有一个独立的程序计数器,各个线程之间的计数器互不影响独立存储,这类内存区是线程私有的。

  (2)此内存区域没有任何OutOfMemoryError的情况。

2.java虚拟机栈

  (1)线程私有,生命周期与线程相同

  (2)描述的是java方法执行的内存模型:每个方法在执行的时候都会创建一个栈帧,用于存储局部变量、操作数栈、动态链接、方法出口等信息,每个方法从调用到执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。

  (3)java内存一般分法:堆内存(heap);栈内存。这里的栈就是虚拟机栈或者说是虚拟机中局部变量的部分。局部变量存放了各种基本的数据类型。局部变量所需的内存空间是在编译期完成分配的

  (4)线程请求深度大于虚拟机所允许的深度,将会抛出StackOverflowError异常;

   如果虚拟机栈可以动态扩展,如果扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常。

3.本地方法栈(Native Method Stack)

  (1)本地方法栈与虚拟机栈所发挥的作用是非常相似的,区别是:虚拟机栈为虚拟机执行java方法服务,本地方法栈则为虚拟机使用到的Native方法服务

4.java堆(java Heap)

  (1)java虚拟机中管理内存中最大的一块。被所有线程贡献的内存区域,在虚拟机启动的时候创建。java堆得唯一目的是存放对象实例,几乎所有的对象实例都在这里分配内存。

  (2)java堆是垃圾收集器管理的主要区域,很多时候被称为GC堆。分代收集算法划分java堆:新生代和老年代,Eden空间、From Survivor空间、To Survivor空间。在java堆中无论怎么划分空间,存放的都是对象实例。对java堆进一步划分的目的是为了更好的回收内存。

  (3)java堆可以处于物理上不连续的内存空间,只要逻辑上连续就可以,当前主流的虚拟机都是可以扩展的。

5.方法区

  (1)各个线程共享的内存区域,用于存储已被虚拟机加载的类信息、常量、静态变量、编辑器编译后的代码等。

  (2)方法区有个别名:Non-Heap(非堆)。不连续内存和可以选择固定大小或者可扩展,还可以不识闲垃圾收集

6.运行时常量池(Runtime COnstant Pool):方法区的一部分

7.直接内存

三、HotSpot虚拟机对象探秘

  1.对象的创建。

    (1)对象的创建过程:首先检查指令的参数是否能在常量池中定位到一个类的符号的引用,并且检查这个符号的引用代表的类是否已被加载、解析和初始化过。如果没有那就先要执行相应的类加载的过程。

    (2)为新生对象分配内存:在类加载完成后,对象所需的内存大小就可以完全确定。内存分配方式是指针碰撞。

  2.对象的内存布局

    (1)对象在内存中的布局分为3块区域:对象头(Header)、实例数据(Instance Data)和对其填充(Padding)

    (2)对象头包括两部分:第一部分是存储对象自身的运行时数据;第二部分时类型指针,虚拟机通过这个指针确定对象是哪个类的实例。

    (3)实例数据是对象真正存储有效信息,即在程序代码中定义的各种类型的字段内容。

  3.对象的访问定位

    (1)目前主流的访问方式是有使用句柄和直接指针两种。

    (2)句柄访问。java堆中会划出一块内存来作为句柄池,reference中存储的就是对象的句柄地址,而句柄中包含了对象实例数据与类型数据各自的具体地址信息。这种方式的好处是reference中存储的是稳定的句柄地址,在对象被移动的时候只会改变句柄中实例数据指针,而reference本身不会修改

四、OutOfMemoryError异常

  1.java堆溢出

    (1)只要不断地创建对象,并且保证GC Roots到对象之间有可达路径来避免垃圾回收机制清除这些对象,那么在对象数量达到最大堆得容量限制之后就会产生内存溢出异常,如下所示的代码中

import java.util.ArrayList;
import java.util.List; public class HeapOOM {
static class OOMObject{}
public static void main(String[] args){
List<OOMObject> list = new ArrayList<>();
while(true){
list.add(new OOMObject());
}
}
}
//java堆内存溢出的时候,异常信息是

Exception in thread "main" java.lang.OutOfMemoryError: Java heap space


    对于上面的问题,要解决堆内存溢出的问题,一般手段是通过内存分析工具堆溢出的堆转储快照进行分析。如果时内存泄漏就找出相应的位置;如果不存在内存的泄漏,就应该检查虚拟机的堆参数,与机器的物理内存看看是否还可以继续调大。

  2.虚拟机栈与本地栈溢出

    (1)栈容量的设置:由-Xss参数设定。

    (2)关于虚拟机栈和本地方法栈有两种异常:

      线程请求的栈深度大于虚拟机所允许的最大深度,就会抛出StackOverflowError异常(栈空间太小还是已经使用的栈空间太大?)

      如果虚拟机在扩展栈的时候无法申请到足够的内存空间,那么会抛出OutOfMemoryErroer异常()

public class JavaVMStackSOF {
private int stackLength=;
public void stackLeak(){
stackLength++;
stackLeak();
}
public static void main(String[] args) throws Throwable{
JavaVMStackSOF oom = new JavaVMStackSOF();
try {
oom.stackLeak();
}catch (Throwable e){
System.out.println("stack length"+oom.stackLength);
} }
}
//在单个线程的情况下,无论是栈帧太大还是虚拟机容量太小,当内存无法分配的时候,虚拟机跑出的都是StackOverflowError

  3.方法区和运行时常量池溢出

  4.本机直接内存溢出

    

    

  

自动内存管理机制之java内存区域与内存溢出异常的更多相关文章

  1. 【iOS开发-35】有了ARC内存管理机制,是否还须要操心内存溢出等问题?——面试必备

    答案:必需要操心啊,ARC也不是万能的. 这里主要是涉及到集合类的数据类型. 比方数组,我们定义了一个可变数组muarr1,然后把一个对象p1加到muarr1中,此时会对这个对象retain一次,相当 ...

  2. Linux内存管理机制简析

    Linux内存管理机制简析 本文对Linux内存管理机制做一个简单的分析,试图让你快速理解Linux一些内存管理的概念并有效的利用一些管理方法. NUMA Linux 2.6开始支持NUMA( Non ...

  3. 从内存管理原理,窥探OS内存管理机制

    摘要:本文将从最简单的内存管理原理说起,带大家一起窥探OS的内存管理机制,由此熟悉底层的内存管理机制,写出高效的应用程序. 本文分享自华为云社区<探索OS的内存管理原理>,作者:元闰子 . ...

  4. JVM自动内存管理机制——Java内存区域(下)

    一.虚拟机参数配置 在上一篇<Java自动内存管理机制——Java内存区域(上)>中介绍了有关的基础知识,这一篇主要是通过一些示例来了解有关虚拟机参数的配置. 1.Java堆参数设置 a) ...

  5. JVM自动内存管理机制——Java内存区域(上)

    一.JVM运行时数据区域概述 Java相比较于C/C++的一个特点就是,在虚拟机自动内存管理机制的帮助下,我们不需要为每一个操作都写像C/C++一样的delete/free代码,所以也不容易出现内存泄 ...

  6. 【深入理解Java虚拟机】自动内存管理机制——内存区域划分

      Java与C++之间有一堵有内存动态分配和垃圾收集技术所围成的"高墙",墙外面的人想进去,墙里面的人却想出来.C/C++程序员既拥有每一个对象的所有权,同时也担负着每一个对象生 ...

  7. 2.1 自动内存管理机制--Java内存区域与内存溢出异常

    自动内存管理机制 第二章.Java内存区域与内存溢出异常 [虚拟机中内存如何划分,以及哪部分区域.什么样代码和操作会导致内存溢出.各区域内存溢出的原因] 一.运行时数据区域 Java虚拟机所管理的内存 ...

  8. 【深入理解Java虚拟机】自动内存管理机制——垃圾回收机制

      Java与C++之间有一堵有内存动态分配和垃圾收集技术所围成的"高墙",墙外面的人想进去,墙里面的人却想出来.C/C++程序员既拥有每一个对象的所有权,同时也担负着每一个对象生 ...

  9. 深入理解Java虚拟机(自动内存管理机制)

    文章首发于公众号:BaronTalk 书籍真的是常读常新,古人说「书读百遍其义自见」还是很有道理的.周志明老师的这本<深入理解 Java 虚拟机>我细读了不下三遍,每一次阅读都有新的收获, ...

随机推荐

  1. Kaggle 网站上的练习

    快速的数据科学教程: https://www.kaggle.com/learn/overview

  2. 关于MySQL 处理重复数据

    统计重复数据 以下我们将统计表中 first_name 和 last_name的重复记录数: mysql> SELECT COUNT(*) as repetitions, last_name, ...

  3. XMLHttpRequest 对象属性参数参考

    readyState:提供当前 HTML 的就绪状态. readyState可能返回的值: 0:请求未初始化(还没有调用 open()). 1:请求已经建立,但是还没有发送(还没有调用 send()) ...

  4. [论文理解] Spatial Transformer Networks

    Spatial Transformer Networks 简介 本文提出了能够学习feature仿射变换的一种结构,并且该结构不需要给其他额外的监督信息,网络自己就能学习到对预测结果有用的仿射变换.因 ...

  5. Android 获取视频照片与刷新媒体库

    1.获取本地所有视频 public void getLoadMedia() { Cursor cursor = UILApplication.instance.getApplicationContex ...

  6. CSS二级菜单

    0.需求:当鼠标hover到按钮上时,出现下拉菜单导航条. 1.问题拆解: (1)HTML应该如何组织比较方便合理 因为题中要求下拉菜单位于按钮的正下方,可以使用列表<li>中嵌套无序列表 ...

  7. Ruby小白入门笔记之<个人记录档>

    书写缘由 快两年的JAVA开发,因为来到一家新公司,产品需要用Ruby开发,故此才有了这从头开始,一入编程深似海啊...... 因为入门时是JAVA,所以理念跟规范早已形成,故此感觉突然采用Ruby编 ...

  8. TCP/IP,Http,Socket的区别

    大学学习网络基础的时候老师讲过,网络由下往上分为物理层.数据链路层.网络层.传输层.会话层.表示层和应用层.通过初步的了解,我知道IP协议对应于网络层,TCP协议对应于传输层,而HTTP协议对应于应用 ...

  9. centos7.5 升级kernel内核版本

    一,查看当前系统内核版本信息 awk -F\' '$1=="menuentry " {print i++ " : " $2}' /etc/grub2.cfg 或 ...

  10. CentOS7 下SaltStack部署

    一,概念SaltStack是一个服务器基础架构集中化管理平台,具备配置管理.远程执行.监控等功能,一般可以理解为简化版的puppet和加强版的func.SaltStack基于Python语言实现,结合 ...