java内存区域/内存溢出汇总

本文主要介绍Java虚拟机中的内存区域与各种内存溢出情况汇总。

数据区域

方法区、堆、虚拟机栈、程序计数器、本地方法栈

方法区

用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码 
运行时常量池:存放编译期生成的字面量和符号引用

异常(OutofMemoryError: PermGen space)产生原因:
1.往常量池中添加大量数据。eg:String.intern()
2.大量的类信息(或者动态代理)

异常处理:
适当增大配置:-XX:PermSize -XX:MaxPermSize

存放对象实例

异常(OutofMemoryError)产生原因:
1.内存泄漏,最后导致内存溢出
2.代码创建过多对象导致超过设置的内存大小

异常处理(参考:http://blog.csdn.net/wisgood/article/details/16818243
1.尽早释放无用对象的引用(好的办法是使用临时变量的时候,让引用变量在退出活动域后自动设置为null,暗示垃圾收集器来收集该对象,防止发生内存泄露。)
2.程序进行字符串处理时,尽量避免使用String,而应使用StringBuffer(因为每一个String对象都会独立占用内存一块区域)
3.尽量少用静态变量(因为静态变量是全局的,GC不会回收)
4.避免集中创建对象尤其是大对象,如果可以的话尽量使用流操作
5. 尽量运用对象池技术以提高系统性能
6.不要在经常调用的方法中创建对象,尤其是忌讳在循环中创建对象
7.优化配置:-Xms -Xmx

1. Q: Java中会存在内存泄漏吗?
    A:  Java中也存在内存泄露。当被分配的对象可达但已无用(未对作废数据内存单元的引用置null)即会引起。

         如:

Java代码

Vector v=new Vector(10);
for (int i=1;i<100; i ) {
    Object o=new Object();
    v.add(o);
    o=null;
}
// 此时,所有的Object对象都没有被释放,因为变量v引用这些对象。
// 对象加入到Vector后,还必须从Vector中删除,最简单释放方法就是将Vector对象设置为null。
2. Q: 内存泄露、溢出的异同? 

    A: 同:都会导致应用程序运行出现问题,性能下降或挂起。

    B: 异:

    1) 内存泄露是导致内存溢出的原因之一;内存泄露积累起来将导致内存溢出。

    2) 内存泄露可以通过完善代码来避免;内存溢出可以通过调整配置来减少发生频率,但无法彻底避免。

虚拟机栈

存放基本数据类型,对象引用,操作数栈,动态链接,方法出口

异常原因:
(StackoverflowError)线程请求的栈深度大于虚拟结所允许的最大深度(eg:递归没有跳出)
(OutOfMemoryError)在扩展栈时无法申请到足够的内存空间

异常处理:
1.检查是否有递归没有正常结束代码
2.适当调整栈内存大小-Xss

程序计数器

当前线程所执行的字节码的行号指示器。(控制代码的执行顺序)

是一块较小的内存空间,没有规定任何OutOfMemoryError

本地方法栈

和虚拟机方法栈类似,只是负责Native方法

HotSpot将此栈与虚拟机栈合二为一。

异常同虚拟机栈异常。
-Xoss修改本地方法栈内存(但HotSpot此参数无效,通过设置-Xss来修改)

本机直接内存

不是虚拟机规范中的内存区域。NIO等一些处理直接使用的内存,可以使用Native函数库直接分配堆外内存,避免Java堆和Native堆中的来回复制数据,

异常现象:
在出现OutOfMemoryError后,Heap Dump 文件中,没有明显的异常,如果程序中又直接或间接的使用的NIO,就可以考虑检查是不是此内存溢出。

异常处理:
-XX:MaxDirectMemorySize

时间: 2024-10-08 21:54:42

java内存区域/内存溢出汇总的相关文章

jvm系列 (一) ---jvm内存区域与溢出

jvm内存区域与溢出 为什么学习jvm 木板原理,最短的一块板决定一个水的深度,当一个系统垃圾收集成为瓶颈的时候,那么就需要你对jvm的了解掌握. 当一个系统出现内存溢出,内存泄露的时候,因为你懂jvm知识,可以更加快速定位错误,可以通过参数去合理设置各内存区域的内存容量. 因为你对jvm的认识,写代码的时候会潜意识地让你注意代码质量,可能你会说是那是小小的性能提升,但是量变会导致质变的. jvm内存区域 jvm内存划分 方法区 虚拟机栈 本地方法栈 堆 程序计数器 程序计数器 当前线程所执行的

java内存区域及溢出异常

内存划分: java虚拟机在执行java程序过程中会把内存分为以下区域进行管理 线程私有的 虚拟机栈 局部变量表 基本数据类型 long和double占用两个slot 对象引用 返回地址 操作数栈 动态链接 方法出口等信息 抛出异常: 栈深度过大 StackOverflowError 申请内存空间不足 OutOfMemoryError 程序计数器 本地方法栈线程共享的 堆 虚拟机启东时创建 方法区 常量池的回收和类型的卸载 运行常量池:字面量和符号引用 翻译出来的直接引用 直接内存 NIO可以使

【深入Java虚拟机】之一:Java内存区域与内存溢出

[深入Java虚拟机]之:Java内存区域与内存溢出 内存区域 Java虚拟机在执行Java程序的过程中会把他所管理的内存划分为若干个不同的数据区域.Java虚拟机规范将JVM所管理的内存分为以下几个运行时数据区:程序计数器.Java虚拟机栈.本地方法栈.Java堆.方法区.下面详细阐述各数据区所存储的数据类型. 程序计数器(Program Counter Register) 一块较小的内存空间,它是当前线程所执行的字节码的行号指示器,字节码解释器工作时通过改变该计数器的值来选择下一条需要执行的

(一)java内存区域与内存溢出

内存模型 java虚拟机在执行java程序的过程中把他所管理的内存划分为若干个不同的数据区域.包括:程序计数器.java虚拟机栈.本地发放栈.java堆.方法区. 内存模型如图:  程序计数器 一块较小的内存空间,它是当前线程所执行的字节码的行号指示器,字节码解释器工作时通过改变该计数器的值来选择下一条需要执行的字节码指令,分支.跳转.循环等基础功能都要依赖它来实现.每条线程都有一个独立的的程序计数器,各线程间的计数器互不影响,因此该区域是线程私有的. 当线程在执行一个Java方法时,该计数器记

深入理解java虚拟机笔记(一)-java内存区域与内存溢出

1. 前言 这是深入理解java虚拟机一书的笔记,来自第二章.因为这本书讲的比较深奥,这是第二次看,需要记录一下笔记. 2. 运行时数据区域 java虚拟机所管理的内存分为以下几个区域. ps:图片来自网络 2.1 程序计数器 程序计数器是一块较小的内存空间,他可以看做是当前线程所执行字节码的行号指示器.字节码解释器工作时就是通过改变这个计数器的值来选去下一条要执行的字节码指令,分之.循环.跳转.异常处理.线程恢复等基础功能都需要依赖这个计数器来完成. 这块内存是线程私有的内存. 如果线程在执行

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

文章主要是阅读<深入理解java虚拟机:JVM高级特性与最佳实践>第二章:Java内存区域与内存溢出异常 的一些笔记以及概括. 好了开始.如果有什么错误或者遗漏,欢迎指出. 一.概述 先上一张图 这张图主要列出了Java虚拟机管理的内存的几个区域. 常有人把Java内存区分为堆内存(Heap)和栈内存(Stack),这种分法比较粗糙,Java内存区域的划分实际上远比这复杂,从上图就可以看出了.堆栈分法中所指的"栈"实际上只是虚拟机栈,或者说是虚拟机栈中的局部变量表部分.接下

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

java内存区域与内存溢出异常 一.运行时数据区域 1.程序计数器:线程私有,用于存储当前所执行的指令位置 2.Java虚拟机栈:线程私有,描叙Java方法执行模型:执行方法时都会创建一个栈帧,存储局部变量,基本类型变量,引用等信息 3.Java本地方法栈:线程私有,为虚拟机使用到的Native方法服务 4.Java堆:线程共享,是垃圾收集器的主要工作地方:存储对象实例等 5.方法区:线程共享:存储类信息,常量,静态变量等 运行时常量:存放编译时生成的各种字面量和符号引用 6.直接内存:机器的内

《深入理解Java虚拟机》读书笔记---第二章 Java内存区域与内存溢出异常

Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想进去,墙里面的人却想出来.这一章就是给大家介绍Java虚拟机内存的各个区域,讲解这些区域的作用,服务对象以及其中可能产生的问题. 1.运行时数据区域 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域. 1.1程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可以看作是当前线程所执行的字节码的行号指示器.在虚拟机的概念模型中里,字

Java内存区域与内存溢出异常-内存区域

Java内存区域与内存溢出异常 概述 对于 C 和 C++程序开发的开发人员来说,在内存管理领域,程序员对内存拥有绝对的使用权,但是也要主要到正确的使用和清理内存,这就要求程序员有较高的水平. 而对于 Java 程序员来说,在虚拟机的自动内存管理机制的帮助下,不再需要为每一个 new 操作去写配对的 delete/free 代码,而且不容易出现内存泄漏和内存溢出问题,看起来由虚拟机管理内存一切都很美好.不过,也正是因为 Java 程序员把内存控制的权力交给了 Java 虚拟机,一旦出现内存泄漏和