java内存区域及溢出异常
内存划分:
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
内存转存储分析工具的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时,返回这个字符串对象,不存在时,添加到常量池并返回字符串的引用。
java内存区域及溢出异常的更多相关文章
- 深入理解java虚拟机系列(一):java内存区域与内存溢出异常
文章主要是阅读<深入理解java虚拟机:JVM高级特性与最佳实践>第二章:Java内存区域与内存溢出异常 的一些笔记以及概括. 好了開始.假设有什么错误或者遗漏,欢迎指出. 一.概述 先上 ...
- 《深入理解Java虚拟机》-----第2章 Java内存区域与内存溢出异常
2.1 概述 对于从事C.C++程序开发的开发人员来说,在内存管理领域,他们即是拥有最高权力的皇帝又是执行最基础工作的劳动人民——拥有每一个对象的“所有权”,又担负着每一个对象生命开始到终结的维护责任 ...
- 《深入理解java虚拟机》第二章 Java内存区域与内存溢出异常
第二章 Java内存区域与内存溢出异常 2.2 运行时数据区域
- 深入了解Java虚拟机(1)java内存区域与内存溢出异常
java内存区域与内存溢出异常 一.运行时数据区域 1.程序计数器:线程私有,用于存储当前所执行的指令位置 2.Java虚拟机栈:线程私有,描叙Java方法执行模型:执行方法时都会创建一个栈帧,存储局 ...
- JVM高级特性与实践(一):Java内存区域 与 内存溢出异常
套用<围城>中的一句话,“墙外面的人想进去,墙里面的人想出来”,用此来形容Java与C++之间这堵内存动态分配和垃圾收集技术所围成的“围墙”就再合适不过了. 对于从事C.C++的开发人员而 ...
- 深入理解java虚拟机---->java内存区域与内存溢出异常
2. java内存区域于内存溢出异常 2.1 概述: 对于C/C++而言,内存管理具有最高的权利,既拥有每一个对象的“所有权”,又担负着每一个对象生命开始到结束的维护责任. 对于java而言,则把内存 ...
- 第二章Java内存区域与内存溢出异常
第二章 Java内存区域与内存溢出异常 一.概述 对与Java程序员来说,在虚拟机自动内存管理机制的帮助下,不再需要为每个new操作去写delete/free代码,不容易出现内存泄露和内存溢出问 题, ...
- 2.1 自动内存管理机制--Java内存区域与内存溢出异常
自动内存管理机制 第二章.Java内存区域与内存溢出异常 [虚拟机中内存如何划分,以及哪部分区域.什么样代码和操作会导致内存溢出.各区域内存溢出的原因] 一.运行时数据区域 Java虚拟机所管理的内存 ...
- java内存区域与内存溢出异常(1)
一. 运行时数据区域 java虚拟机在执行Java程序的过程中,会把它所管理的内存划分为若干个不同的数据区域偶 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 ...
- vue初级 总结
mvvm m:代表 data v 代表 view vm 代表 Vue 的实例 v-cloak 指令 解决闪烁的问题 需要在 style 标签中加入 [v-cloak];{ display:none } ...
- spark 笔记 1: 如何着手
必读:从官方的开发者页面着手,包括如何构建spark以及编码规范(强烈建议读读编程规范)等:https://cwiki.apache.org/confluence/display/SPARK/Cont ...
- Centos6 源码安装mysql5.6
这里介绍如何使用centos6.*来安装mysql5.6版本. 先做一下准备工作 确定好用于运行mysql的用户,安全起见,建议拒绝次用户登录,仅用于运行程序. useradd mysql -s/sb ...
- Hidden的应用
在写jsp中如果一个 请求的参数(例如:paramTypeCode)不能在另一个请求中使用,我们为了能让他在请求中使用可以利用隐藏域来表示,下面介绍他的用法: 1 <input type= ...
- Dark 运算符
1 赋值运算符 ??= b ??= "hello" 若b为空,则把"hello"赋给b . 2 三目运算符 boolean ? expression1 : ex ...
- MobileNetV2: Inverted Residuals and Linear Bottlenecks
1. 摘要 作者提出了一个新的网络架构 MobileNetV2,该架构基于反转残差结构,其中的跳跃连接位于较瘦的瓶颈层之间.中间的扩展层则利用轻量级的深度卷积来提取特征引入非线性,而且,为了维持网络的 ...
- electron-Menu创建原生应用菜单和上下文菜单。
当在MacOS.Windows.Linux中使用menu设置程序菜单时,会设置在各个程序窗体的顶层. Note: 如果没有在app中设置一个菜单,系统会自动生成一个默认菜单, 默认生成的菜单中包含了一 ...
- Kubernetes 中文文档
Kubernetes 中文文档 如果想学习 Kubernetes 的小伙伴,可以参考如下文档学习: https://www.kubernetes.org.cn/docs 文档中详细讲解了 k8s 的设 ...
- Appium - multiprocessing.pool.MaybeEncodingError-【 “Can’t pickle local object ‘PoolManager.__init__.<locals>.<lambda>‘】
公司同事学习自动化新装环境后,run多进程测试用例时出错: multiprocessing.pool.MaybeEncodingError: Error sending result: ’<ap ...