内存划分:

java虚拟机在执行java程序过程中会把内存分为以下区域进行管理

线程私有的
  虚拟机栈
    局部变量表
      基本数据类型
        long和double占用两个slot
      对象引用
      返回地址

    操作数栈
    动态链接
    方法出口等信息

    抛出异常:

      栈深度过大 StackOverflowError
      申请内存空间不足 OutOfMemoryError

  程序计数器

  本地方法栈
线程共享的
  堆
    虚拟机启东时创建
  方法区
    常量池的回收和类型的卸载
    运行常量池:字面量和符号引用 翻译出来的直接引用

  直接内存

    NIO可以使用Native函数库直接分配,这样能显著增加效率,因为不用在java堆和native堆中复制数据

    然而,分配内存时如果直接内存加java内存超过计算机物理内存限制,就会报出OutOfMemoryError

虚拟机对象:
对象的创建:
  1.new定义到常量池中的一个符号
  2.检查符号代表的类是否被装载 解析 初始化
    如果没有,加载类
  3.加载完类后,为新生对象分配内存(类加载后会确定分配的内存大小)

    内存分配方法:
      如果内存是规整的 指针碰撞(指针移动对象的内存大小的位置)
      如果内存是交错的 空闲列表 列表记录哪些内存是可用的,分配对象实例时分配给足够大的内存给对象,并更新列表

      选择哪种分配方法由java堆是否规整来决定,堆是否规整由java回收器是否采用压缩整理功能决定

    分配内存的线程安全问题
      对分配内存的动作进行同步处理
      TLAB 本地分配缓冲区 每个线程预先分配一块内存,哪个线程要分配内存,就在哪个线程上面分配,只有在TLAB用尽并分配新的TLAB时才需要同步锁
  4.内存分配之后,内存空间都初始化为零值
  5.对对象进行必要设置,设置对象头(类 元数据 hashcode gc年代)

对象的布局:
  对象头
    运行时数据
      HashCode GC分代年龄 锁状态标识 线程持有的锁 偏向线程id 偏向时间戳
    类型指针
      如果是数组,对象头要保存记录数组的长度信息
  实例数据
    虚拟机分配策略
  对齐填充
    hotpot虚拟机需要对象的大小是8个字节的整数倍 对象头正好是8个字节的倍数 如果实例数据没有对齐时,需要用对齐填充来补充

对象的访问定位:
  引用定义在栈上,虚拟机规范没有定义引用以何种方式定位对象位置,由虚拟机自身来完成,主流访问方式两种:
  1.句柄

    

    好处:对象移动时,只需要改变句柄的实例数据指针,不需要改变栈中本地变量的指针(它指向)

  2.直接指针

    

    好处 速度更快,节省一次指针定位的时间

OutOfMemoryError
java堆溢出:

  通过-XX:+HeapDumpOnOutOfMemoryError可以在虚拟机异常时候dump出当前堆转储快照以便事后分析
  VM Args:-Xms20m -Xmx20m -XX:+HeapDumpOutOfMemoryError

  MAT Tool插件安装

  内存转存储分析工具的Eclipse Memory Analyzer的使用

/**
*
*/
package com.gengsc.oom; /**
* VM args:-Xss2M
*
* @author shichaogeng
*
* 2017年6月26日
*/
public class JavaVMOOM { private void dontStop() {
while (true) {
//...
}
} public void StackLeakByThread() {
while (true) {
Thread thread = new Thread(new Runnable() {
public void run() {
dontStop();
}
});
thread.start();
}
} public static void main(String[] args) {
JavaVMOOM oom = new JavaVMOOM();
oom.StackLeakByThread();
}
}

通过不断的建立线程可以导致OutOfMemoryError

这种异常很奇特,为每个线程分配的内存越大,反而越容易产生内存溢出

这是什么原因呢,操作系统为每个进程分配内存是有限制的,减去Xms,减去MaxPermSize,剩下的就分给虚拟机栈和本地方法栈了,每个线程分配内存越大,线程数量越少,越易产生内存溢出异常

正确的处理方式就是减少java堆的大小和栈内存来换取线程数量。

方法区域和运行时常量池内存溢出:

String.intern()方法:当常量池中存在string时,返回这个字符串对象,不存在时,添加到常量池并返回字符串的引用。

关于String#intern()的详细内容可以点我

java内存区域及溢出异常的更多相关文章

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

    文章主要是阅读<深入理解java虚拟机:JVM高级特性与最佳实践>第二章:Java内存区域与内存溢出异常 的一些笔记以及概括. 好了開始.假设有什么错误或者遗漏,欢迎指出. 一.概述 先上 ...

  2. 《深入理解Java虚拟机》-----第2章 Java内存区域与内存溢出异常

    2.1 概述 对于从事C.C++程序开发的开发人员来说,在内存管理领域,他们即是拥有最高权力的皇帝又是执行最基础工作的劳动人民——拥有每一个对象的“所有权”,又担负着每一个对象生命开始到终结的维护责任 ...

  3. 《深入理解java虚拟机》第二章 Java内存区域与内存溢出异常

    第二章 Java内存区域与内存溢出异常 2.2 运行时数据区域  

  4. 深入了解Java虚拟机(1)java内存区域与内存溢出异常

    java内存区域与内存溢出异常 一.运行时数据区域 1.程序计数器:线程私有,用于存储当前所执行的指令位置 2.Java虚拟机栈:线程私有,描叙Java方法执行模型:执行方法时都会创建一个栈帧,存储局 ...

  5. JVM高级特性与实践(一):Java内存区域 与 内存溢出异常

    套用<围城>中的一句话,“墙外面的人想进去,墙里面的人想出来”,用此来形容Java与C++之间这堵内存动态分配和垃圾收集技术所围成的“围墙”就再合适不过了. 对于从事C.C++的开发人员而 ...

  6. 深入理解java虚拟机---->java内存区域与内存溢出异常

    2. java内存区域于内存溢出异常 2.1 概述: 对于C/C++而言,内存管理具有最高的权利,既拥有每一个对象的“所有权”,又担负着每一个对象生命开始到结束的维护责任. 对于java而言,则把内存 ...

  7. 第二章Java内存区域与内存溢出异常

    第二章 Java内存区域与内存溢出异常 一.概述 对与Java程序员来说,在虚拟机自动内存管理机制的帮助下,不再需要为每个new操作去写delete/free代码,不容易出现内存泄露和内存溢出问 题, ...

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

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

  9. java内存区域与内存溢出异常(1)

    一. 运行时数据区域 java虚拟机在执行Java程序的过程中,会把它所管理的内存划分为若干个不同的数据区域偶 1.程序计数器 程序计数器是一块较小的内存空间,作用是当前线程所执行的字节码的行号指示器 ...

随机推荐

  1. The control collection cannot be modified during DataBind, Init, Load, PreRender or Unload phases.

    https://stackoverflow.com/questions/5508666/dynamically-add-html-to-asp-net-page https://stackoverfl ...

  2. vue初级 总结

    mvvm m:代表 data v 代表 view vm 代表 Vue 的实例 v-cloak 指令 解决闪烁的问题 需要在 style 标签中加入 [v-cloak];{ display:none } ...

  3. spark 笔记 1: 如何着手

    必读:从官方的开发者页面着手,包括如何构建spark以及编码规范(强烈建议读读编程规范)等:https://cwiki.apache.org/confluence/display/SPARK/Cont ...

  4. Centos6 源码安装mysql5.6

    这里介绍如何使用centos6.*来安装mysql5.6版本. 先做一下准备工作 确定好用于运行mysql的用户,安全起见,建议拒绝次用户登录,仅用于运行程序. useradd mysql -s/sb ...

  5. Hidden的应用

    在写jsp中如果一个 请求的参数(例如:paramTypeCode)不能在另一个请求中使用,我们为了能让他在请求中使用可以利用隐藏域来表示,下面介绍他的用法: 1    <input type= ...

  6. Dark 运算符

    1 赋值运算符 ??= b ??= "hello" 若b为空,则把"hello"赋给b . 2 三目运算符 boolean ? expression1 : ex ...

  7. MobileNetV2: Inverted Residuals and Linear Bottlenecks

    1. 摘要 作者提出了一个新的网络架构 MobileNetV2,该架构基于反转残差结构,其中的跳跃连接位于较瘦的瓶颈层之间.中间的扩展层则利用轻量级的深度卷积来提取特征引入非线性,而且,为了维持网络的 ...

  8. electron-Menu创建原生应用菜单和上下文菜单。

    当在MacOS.Windows.Linux中使用menu设置程序菜单时,会设置在各个程序窗体的顶层. Note: 如果没有在app中设置一个菜单,系统会自动生成一个默认菜单, 默认生成的菜单中包含了一 ...

  9. Kubernetes 中文文档

    Kubernetes 中文文档 如果想学习 Kubernetes 的小伙伴,可以参考如下文档学习: https://www.kubernetes.org.cn/docs 文档中详细讲解了 k8s 的设 ...

  10. Appium - multiprocessing.pool.MaybeEncodingError-【 “Can’t pickle local object ‘PoolManager.__init__.<locals>.<lambda>‘】

    公司同事学习自动化新装环境后,run多进程测试用例时出错: multiprocessing.pool.MaybeEncodingError: Error sending result: ’<ap ...