转载于:http://www.itxuexiwang.com/a/liunxjishu/2016/0205/75.html
文章目录
1. Java内存区域与内存溢出异常
1.1. 运行时数据区域
1.1.1. 程序计数器
1.1.2. java虚拟机栈
1.1.3. 本地方法栈
1.1.4. Java堆(Java Heap)
1.1.5. 方法区
1.1.6. 运行时常量池
1.1.7. 直接内存
1.2. HotSpot虚拟机
1.2.1. 对象的创建
1.2.2. 对象的访问定位#p#分页标题#e#
1.3. OOM异常的解决思路
1.4. 参考
深入理解jvm之内存区域与内存溢出
Java内存区域与内存溢出异常
运行时数据区域
深入理解jvm之内存区域与内存溢出
程序计数器
当前线程所执行的字节码的行号指示器
当前线程私有
不会出现OutOfMemoryError情况
java虚拟机栈
#p#分页标题#e#
线程私有,生命周期与线程相同
java方法执行的内存模型,每个方法执行的同时都会创建一个栈帧,存储局部变量表(基本类型、对象引用)、操作数栈、动态链接、方法出口等信息
StackOverflowError异常:当线程请求的栈深度大于虚拟机所允许的深度
OutOfMemoryError异常:如果栈的扩展时无法申请到足够的内存
本地方法栈
与虚拟机栈相似,主要为虚拟机使用到的Native方法服务,在HotSpot虚拟机中直接把本地方法栈与虚拟机栈二合一
Java堆(Java Heap)
java堆是被所有线程共享的一块内存区域,在 虚拟机启动时创建。此区域的唯一目的就是存储对象实例。java堆是垃圾收集器管理的主要区域。java堆还可以细分为:新生代与老年代。在细一点有 Eden空间、Form Survivor空间、To Survivor空间等。
可以通过-Xmx和-Xms控制堆的大小
#p#分页标题#e#
OutOfMemoryError异常:当在堆中没有内存完成实例分配,且堆也无法再扩展时。
方法区
线程间共享
用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据
OutOfMemoryError异常:当方法区无法满足内存的分配需求时
运行时常量池
方法区的一部分
用于存放编译期生成的各种字面量与符号引用
OutOfMemoryError异常:当常量池无法再申请到内存时
#p#分页标题#e#直接内存
NIO可以使用Native函数库直接分配堆外内存,堆中的DirectByteBuffer对象作为这块内存的引用进行操作
大小不受Java堆大小的限制,受本机(服务器)内存限制
OutOfMemoryError异常:系统内存不足时
HotSpot虚拟机
对象的创建
虚拟机遇到一条new指令时,首先将去检查这个对象的参数是否在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有,必须先执行类的加载过程。
在类加载检查通过后,虚拟机将为新生对象分配内存。对象所需内存大小再类加载完成后便可确定。内存分配可以采用“指针碰撞”与“空闲列表”的方式。
对象的访问定位
java程序需要通过栈上的reference数据来操作堆上的具体对象。访问方式有使用句柄和直接指针两种。
#p#分页标题#e#
句柄访问 java堆中将会划分出一块内存来作为句柄池,reference中存储的就是对象的句柄地址,而句柄中包含了对象实例数据与类型数据各自的具体地址信息
直接指针访问 java堆对象的布局中必须考虑如何放置访问类型数据的相关信息,reference中存储的就是对象地址
OOM异常的解决思路
生成Dump快照文件:
通过jvm参数—XX:-HeapDumpOnOutOfMemoryError可以让JVM在出现内存溢出是Dump出当前的内存转储快照
用jmap生产dump文件,win通过任务管理器查看tomcat的进程pid,linux用ps命令查看进程pid,然后用jmap命令
先通过内存映像分析工具(如Eclipse的Memory Analyzer)进行分析,常见的情况有:
内存泄露,对象已经死了,无法通过垃圾收集器进行自动回收,通过找出泄露的代码位置和原因,才好确定解决方案;
内存溢出,内存中的对象都还必须存活着,这说明Java堆分配空间不足,检查堆设置大小(-Xmx与-Xms),检查代码是否存在对象生命周期太长、持有状态时间过长的情况。#p#分页标题#e#
OOM异常示例:
package oom;
import java.util.ArrayList;
import java.util.List;
/** #p#分页标题#e#
* VM Args: -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError
* @ClassName: HeapOOM
*
*/
public class HeapOOM {
#p#分页标题#e# static class OOMObject{
}
public static void main(String[] args) {
List<OOMObject> list = #p#分页标题#e#new ArrayList<OOMObject>();
while(true){
list.add(new OOMObject());
}
}
}
分享到:
相关推荐
深入理解JVM内存区域与内存溢出异常
了解JVM内存管理和G1 GC的工作原理,对于优化Java应用程序性能、避免内存溢出等问题至关重要。开发者可以通过调整JVM参数,如设置年轻代与老年代的比例、分配的Region数量、暂停时间目标等,来优化G1 GC的行为,从而...
2019最新深入理解JVM内存结构及运行原理(JVM调优)高级核心课程视频教程下载。JVM是Java知识体系中的重要部分,对JVM底层的了解是每一位Java程序员深入Java技术领域的重要因素。本课程试图通过简单易懂的方式,系统...
### Java虚拟机(JVM)内存设置与调优详解 ...通过对JVM内存管理机制的理解,结合具体的应用场景,开发者可以有效地避免内存溢出错误,确保应用稳定高效地运行。希望本文能为您的Java项目提供有力的技术支持。
通过本实验,旨在深入理解JVM内存管理机制以及各种内存区域的特点,并通过具体的编程实践来触发并分析这些异常,进而提升对Java应用程序性能调优和故障排查的能力。 #### 实验目标 1. **理解内存区域与内存区域...
【JVM内存溢出】指的是Java...总的来说,解决JVM内存溢出问题需要深入了解Java内存模型,合理配置JVM参数,并结合性能监控工具进行诊断和优化。这不仅可以避免服务器因内存问题崩溃,还能提升整体系统的稳定性和性能。
【初探JVM内存区域】 Java虚拟机(JVM)是Java编程语言的核心组成部分,它为Java应用程序提供了运行环境。理解JVM内存区域对于优化Java应用性能至关重要。本篇文章将详细探讨JVM中的主要内存区域及其作用。 1. **...
本文将深入探讨如何在Java中获取JVM内存大小,包括堆内存的总量、最大值以及剩余空间,并解析给定代码片段中的关键概念。 ### JVM内存模型 在讨论如何获取JVM内存大小之前,首先需要理解JVM的内存布局。JVM内存...
在“jvm的内存结构图的ppt模型分析”中,我们将深入探讨JVM内存的不同区域及其功能。 首先,JVM内存可以分为堆内存和栈内存两大主要部分,它们都是线程共享的。 1. **堆内存**:这是Java应用中所有对象实例的存储...
2. **内存区域划分**:JVM内存分为堆内存和栈内存,其中栈内存又包括方法区、虚拟机栈、本地方法栈。堆内存用于存储对象实例,而栈内存则存储线程私有的局部变量、方法参数和运算结果。方法区存储类的信息,如常量池...
在Java环境中,JVM负责内存的分配和回收,因此对JVM的深入理解至关重要。 在Java应用中,内存主要分为以下几个区域: 1. **堆内存**:用于存储对象实例,是所有线程共享的一块内存区域。当堆内存不足时,可能会引发...
3. **运行时数据区**:JVM内存分为堆、栈、方法区、程序计数器和本地方法栈五个区域。堆存储对象实例,栈处理方法调用,方法区存储类信息,程序计数器记录下一条指令地址,本地方法栈服务Native方法。 4. **内存...
JVM内存模型与垃圾回收是...总的来说,理解JVM内存模型和垃圾回收机制对于优化Java应用性能至关重要,它涉及到内存分配策略、垃圾收集算法的选择以及内存参数的调整,这些都需要开发者具备深入的JVM知识和实践经验。
通过以上内容可以看出,深入理解JVM内存管理和垃圾收集机制对于提高应用程序的性能至关重要。尽管现代JVM已经非常智能,能够自动管理大部分内存相关的工作,但在面对复杂的应用场景时,仍然需要开发人员具备一定的...
标题中的“关于tomcat乱码以及tomcat jvm 内存溢出问题的解决方案和理论”涉及了两个关键的IT概念:Tomcat服务器的字符编码问题和Java虚拟机(JVM)内存管理的问题。让我们逐一深入探讨这两个主题。 首先,我们来...
### JVM内存结构详解 #### 一、概述 Java虚拟机(JVM)作为Java程序的运行环境,其核心组件之一便是内存管理系统。...希望本文能帮助读者深入理解JVM的核心概念,并在实践中运用这些知识来提升Java应用的表现。
本资料总结主要关注JVM内存分配及其运行原理,这对于理解和优化Java应用程序的性能至关重要。 1. **JVM内存结构** JVM内存分为几个关键区域:方法区(Method Area)、堆(Heap)、栈(Stack)、程序计数器(PC ...
JVM的内存结构是Java开发者需要了解的基础知识之一,了解JVM的内存结构可以帮助开发者更好地理解Java语言的执行机制,提高程序的执行效率和性能。 在JVM中,字符串常量池的实现是通过String.intern()方法来实现的,...
JVM(Java Virtual Machine)是Java程序运行的基础,它负责解析和执行字节码,为Java应用程序提供了一个跨平台的运行环境。...通过不断学习和实践,我们可以更深入地了解JVM,从而编写出更加高效、可靠的Java程序。