Java内存泄漏定位
Java虚拟机内存分为五个区域:方法区,堆,虚拟机栈,本地方法栈,程序计数器。其中方法区和堆是java虚拟机共享的内存区域,虚拟机栈,本地方法栈,程序计数器是线程私有的。
程序计数器(Program Counter Register):
当前线程执行字节码的行号指示器。通过改变这个指示器的值来选取下一条需要执行的字节码指令。这个内存区域是Java虚拟机唯一一个没有定义OutOfMemeryError情况的区域。
Java虚拟机栈(Java Visual Machine Stacks):
虚拟机栈描述的是Java方法执行的内存模型:每个方法执行是都会创建栈帧(Stack Frame)用于存储局部变量,操作栈,方法信息,动态链接,方法出口等信息。
在java虚拟机规范中,对于这两个区域规定了两种情况的异常:1)如果线程请求的栈深度大于虚拟机所允许的深度将会抛出StackOverFlowError异常, 2)Java虚拟机可以动态扩展,当无法申请到足够的内存时会抛出OutOfMemeryError
本地方法栈(Native Method Stacks)
本地方法栈与Java虚拟机栈非常类似,其区别是Java虚拟机栈为虚拟机执行Java方法服务,而本地方法栈是虚拟机使用到的Native方法服务。
所以本地方法栈也可能出现两种与Java虚拟机栈相同的异常。
Java堆(Java Heap)
Java堆是Java虚拟机管理的最大的一块内存区域,java堆是被所有Java线程共享的,在Java虚拟机启动时创建,此内存的唯一目的就是存放对象实例。几乎所有的对象实例都要分配在堆中。(随着JIT编译器的发展,逃逸分析技术的逐渐成熟,栈上分配,标量替换等优化技术,使得部分对象不再分配在堆上。)
Java堆的大小通过 -Xmx和-Xms两个参数控制。但是当堆的内存再无法扩展时,就会出现OutOfMemeryError。
方法区(Method Area)
方法区与Java堆一样,是各个线程共享的内存区域,他用于存储类信息,常量,静态变量以及及时编译后的代码等数据。当方法区无法满足内存分配需求时,将抛出OutOfMemeryError.
说了Java虚拟机中分为五个区域,并且也知道了在Java程序计数器区域不会出现OOM(OutOfMemeryError),那么下面就对除了程序计数器以外的四个区域出现OOM的原理以及解决方式进行讲解
1.Java虚拟机栈与本地方法栈
栈的大小控制参数时 -Xss。
Java虚拟机在栈中定义了两种异常,StrackOverFlowError和OutOfMemeryError。当请求栈的深度大于java虚拟机所允许的最大深度则抛出StrackOverFlowError;如果Java虚拟机在栈扩展时,没有申请到足够的空间时,则抛出OutOfMemeryError。
StrackOverFlowError:Java虚拟机在运行中,调用方法时,都要创建栈帧,当栈的空间不够时就会产生StrackOverFlowError。那么对应的解决方法就只能是调节-Xss参数,或者减少方法的调用,减小栈帧的大小两种方式。
OutOfMemeryError:在栈上出现OOM一般是多线程的情形。首先咋们解析一下栈使用的空间可以有多大,拿32位操作系统来举例, 最大内存2G - Xmx(最大堆容量)- MaxPermSize(最大方法区容量)- 虚拟机本身耗费的内存和程序计数器使用的内存。 剩下的内存就是栈可以使用的空间,当Xss配置的参数一定时,那么在不断的创建线程过程中,遇到不能申请到栈空间的时候就会抛出OOM,那么对应的解决方式就是,调节-Xss参数降低栈大小,或者调节-Xmx以及MaxPermSize的大小扩大留给栈的空间。
2.方法区内存溢出
方法区的大小通过-PermSize和-MaxPermSize控制。
因为类常量和运行时常量也存储在方法区中,所以运行时常量过多也可导致方法区的OOM,但是没有直接控制常量池大小的参数,只能通过-PermSize和-MaxPermSize来间接控制。
在Spring以及Hibernate,Mybatis中都会使用GeneratedConstructorAccessor、动态代理以及CGLib字节码增强技术的等动态生成类,那么就需要强大的方法区来支撑。
3.堆内存的溢出
堆内存的溢出比较复杂,需要调节GC等多种参数,我们在后面的章节中会进行讲解。
在上一节中Java 出现内存溢出的定位以及解决方式 中对于Java虚拟机栈以及方法区的内存出现的异常以及处理方式进行了解析,因为Java虚拟机对于堆的管理十分复杂,而且Java虚拟机中最主要的内存区域,所以单独提出一节进行分析。
先来解释一下对象存活??
什么样的对象是已经死了的对象,需要垃圾回收器进行回收,这个概念至关重要,因为它影响到垃圾回收器对于哪一个对象进行回收。可以从GCRoot访问到的对象是存活的对象,那么以外的对象就是已死的对象。
GCRoot:包括四种 1)Java虚拟机栈中存放的reference指针指向的对象 2)本地方法栈中reference指向的对象 3)方法区中的常量引用对象 4)方法区中静态类属性引用的对象
GC 垃圾回收器
垃圾回收器采用的是标记-清除-整理算法回收内存。
Minor GC和Full GC,Minor GC是对年轻代回收的过程, Full GC 是对整个堆以及方法区进行回收的过程。
分代收集,在Java虚拟机中对于堆的内存区域进行再划分为,Young Generation(年轻代)和Old Generation(老代)
年轻代用于存储年龄没有达到-XX:PretenureSizeThreshold的对象,老代用于存储年龄超过了-XX:PretenureSizeThreshold的对象。
然后对于年轻代再进行细分,Eden Space, From Space, ToSpace三个内存区域,这三个内存区域的大小由-XX:SurivorRadio来定义。在进行Minor GC的时候,会将Eden Space和From Space中存活的对象Copy到ToSpace中,并将超过年龄的对象Copy到老代,当老代空间不足时,启动Full GC。
(上面概念只是进行了简单的描述,在JVM实际运行过程中要复杂的多,不过大概原理是这样的)。
了解了上面基本概念之后,看下面的垃圾回收器就比较简单了。垃圾回收器总共分为七种:Serial收集器与Serial Old 收集器,Parallel Scavenge 收集器与Parallel Old,ParNew收集器,CMS收集器,G1(Garbage First)收集器。
1.Serial收集器与Serial Old 收集器,分别用于收集年轻代和老年代内存区域,Serial是单线程的垃圾收集器,在运行过程中需要暂停所有的Java线程。
2.ParNew收集器实际上就是Serial收集器的多线程版本(针对于年轻代)。
3.Parallel Scavenge 收集器与Parallel Old: 是并行的多线程垃圾处理器,可以规定最大垃圾收集停顿时间-XX:MaxGCPauseMillis以及设置吞吐量大小-XX:GCTimeRadio.
4.CMS 收集器(Concurrent Mark Sweep): 是针对于老年代的多线程并发执行的垃圾回收器。以获取最短回收停顿时间为目标的收集器(主要用于B/S架构的服务器上)
5. G1(Garbage First)收集器: 是最新的垃圾回收器,适合于(JDK1.6_update14)以上的JVM;G1将整个Java堆(包括新生代和老年代)划分为多个固定大小的独立区域,并且跟踪这些区域里面的垃圾堆积程度,在后台维护一个垃圾优先列表,每次根据允许收集的时间,优先回收垃圾最多的区域。
总之,要想获得最大吞吐量的服务类型就采用Parallel Scavenge 收集器与Parallel Old收集器组合。要实现实时系统最好采用CMS 收集器(Concurrent Mark Sweep)。不过客户端由于比较小,还是使用Serial比较好。
Java内存泄漏定位的更多相关文章
- java内存泄漏的定位与分析
1.为什么会发生内存泄漏 java 如何检测内在泄漏呢?我们需要一些工具进行检测,并发现内存泄漏问题,不然很容易发生down机问题. 编写java程序最为方便的地方就是我们不需要管理内存的分配和释放, ...
- (转)java内存泄漏的定位与分析
转自:http://blog.csdn.net/x_i_y_u_e/article/details/51137492 1.为什么会发生内存泄漏 java 如何检测内在泄漏呢?我们需要一些工具进行检测, ...
- Java内存泄漏分析系列之一:使用jstack定位线程堆栈信息
原文地址:http://www.javatang.com 前一段时间上线的系统升级之后,出现了严重的高CPU的问题,于是开始了一系列的优化处理之中,现在将这个过程做成一个系列的文章. 基本概念 在对J ...
- Java内存泄漏分析系列之五:常见的Thread Dump日志案例分析
原文地址:http://www.javatang.com 症状及解决方案 下面列出几种常见的症状即对应的解决方案: CPU占用率很高,响应很慢 按照<Java内存泄漏分析系列之一:使用jstac ...
- 如何排查Java内存泄漏?看完我给跪了!
没有经验的程序员经常认为Java的自动垃圾回收完全使他们免于担心内存管理.这是一个常见的误解:虽然垃圾收集器做得很好,但即使是最好的程序员也完全有可能成为严重破坏内存泄漏的牺牲品.让我解释一下. 当不 ...
- java内存泄漏的几种情况
转载于http://blog.csdn.net/wtt945482445/article/details/52483944 Java 内存分配策略 Java 程序运行时的内存分配策略有三种,分别是静态 ...
- java内存泄漏
java内存泄漏主要分成两个方面: (1)堆中申请的空间没有被释放 (2)对象已不在被使用,但是仍然存在在内存当中 以下集中情况可能会导致内存泄漏 (1)静态集合的使用hashmap和vector,静 ...
- Java内存泄漏分析与解决方案
Java内存泄漏是每个Java程序员都会遇到的问题,程序在本地运行一切正常,可是布署到远端就会出现内存无限制的增长,最后系统瘫痪,那么如何最快最好的检测程序的稳定性,防止系统崩盘,作者用自已的亲身经历 ...
- Java内存泄漏分析系列之二:jstack生成的Thread Dump日志结构解析
原文地址:http://www.javatang.com 一个典型的thread dump文件主要由一下几个部分组成: 上图将JVM上的线程堆栈信息和线程信息做了详细的拆解. 第一部分:Full th ...
随机推荐
- atom插件安装失败解决方法
在atom 上下载插件失败,可以用下面的方法. 1.找到C:/Users/你的用户名/.atom/packages/文件夹内 2.在.atom packages 目录下 使用gitbash 3.git ...
- 在云服务器跑Python程序
最近在鼓弄这TensorFlow的模型,有些模型实在是太大了,CPU占用率100%不说,还一跑起来就跑个大半天,严重影响了学习的进度,所以由于手里刚有一个不大使用的云服务器,配置虽然不咋地,至少还能跑 ...
- Java生成带logo二维码
目前生成二维码的方式有很多种,本例采用谷歌的zxing,去白边,添加logo等处理均在代码中有注释 demo连接 https://github.com/littlechaser/qrcode.git
- 二、fread与fwrite
fread 原型:size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream); 参数: ptr:数据存放地址 size:一个对象的 ...
- 利用ASK/OOK 发射模块,实现信号重放
本文以打开无线控制的电动车库卷帘门为目标,深入研究了ASK/OOK的编/解码,并用树莓派+五元钱的ASK/OOK 发射模块 背景 车库装了电动卷帘门,为了了解其安全性,也是为了能自主控制,研究了下其遥 ...
- vue导出excel
1.按装依赖 cnpm install -S file-saver xlsx cnpm install -D script-loader 2.引入Blob.js和expor2Excal.js 3.在m ...
- JS实现数组的相减操作
/** * 数组相减的方法 * @param {Array} a * @param {Array} b */ function arrSubtraction(a, b) { if (!a || !b ...
- html 刷新重载方法汇总
一.javascript页面刷新重载的方法: <a href="javascript:location.reload();">点击重新载入页面</a> &l ...
- seajs引入jquery框架问题
seajs引入jquery框架时出现的问题 原因:由于seajs是遵循cmd规范,而jquery是遵循amd规范,所以在seajs调用jquery框架时,需要将amd转换为cmd 转换方法:jquer ...
- nginx的变量参数 详解
$args #请求中的参数值 $query_string #同 $args $arg_NAME #GET请求中NAME的值 $is_args #如果请求中有参数,值为"?",否则为 ...