Java OOM 常见情况

摘要:
Java中的内存泄漏与C++中的忘记删除不同。这通常是由于逻辑原因。根据JVM规范,除了程序计数器不会抛出OOM之外,其他内存区域可能会抛出OOM。最常见的OOM案例有以下三种:java。lang.OutOfMemoryError:Javaheapspace------˃Java堆内存溢出。这是最常见的情况,通常由内存泄漏或堆大小设置不当引起。Java语言lang.OutOfMemoryError:PrmGenspace-----˃Java永久生成溢出,即方法区域溢出,通常发生在大量的类或jsp页面中,或者使用cglib和其他反射机制中,因为上述情况会生成存储在方法区域中的大量类信息。Java.lang.StackOverflowError-------˃不会引发OOMerror,但它也是一种常见的Java内存溢出。
Java OOM 常见情况

原文:https://blog.csdn.net/qq_42447950/article/details/81435080 

1)什么是OOM? 

OOM,全称“Out Of Memory”,翻译成中文就是“内存用完了”,来源于java.lang.OutOfMemoryError。看下关于的官方说明: Thrown when the Java Virtual Machine cannot allocate an object because it is out of memory, and no more memory could be made available by the garbage collector. 意思就是说,当JVM因为没有足够的内存来为对象分配空间并且垃圾回收器也已经没有空间可回收时,就会抛出这个error(注:非exception,因为这个问题已经严重到不足以被应用处理)。

2)为什么会OOM?

为什么会没有内存了呢?原因不外乎有两点:

1)分配的少了:比如虚拟机本身可使用的内存(一般通过启动时的VM参数指定)太少。

2)应用用的太多,并且用完没释放,浪费了。此时就会造成内存泄露或者内存溢出。

内存泄露:申请使用完的内存没有释放,导致虚拟机不能再次使用该内存,此时这段内存就泄露了,因为申请者不用了,而又不能被虚拟机分配给别人用。

内存溢出:申请的内存超出了JVM能提供的内存大小,此时称之为溢出。

在之前没有垃圾自动回收的日子里,比如C语言和C++语言,我们必须亲自负责内存的申请与释放操作,如果申请了内存,用完后又忘记了释放,比如C++中的new了但是没有delete,那么就可能造成内存泄露。偶尔的内存泄露可能不会造成问题,而大量的内存泄露可能会导致内存溢出。

而在Java语言中,由于存在了垃圾自动回收机制,所以,我们一般不用去主动释放不用的对象所占的内存,也就是理论上来说,是不会存在“内存泄露”的。但是,如果编码不当,比如,将某个对象的引用放到了全局的Map中,虽然方法结束了,但是由于垃圾回收器会根据对象的引用情况来回收内存,导致该对象不能被及时的回收。如果该种情况出现次数多了,就会导致内存溢出,比如系统中经常使用的缓存机制。Java中的内存泄露,不同于C++中的忘了delete,往往是逻辑上的原因泄露。

3)OOM的类型

JVM内存模型:

按照JVM规范,JAVA虚拟机在运行时会管理以下的内存区域:

  • 程序计数器:当前线程执行的字节码的行号指示器,线程私有
  • JAVA虚拟机栈:Java方法执行的内存模型,每个Java方法的执行对应着一个栈帧的进栈和出栈的操作。
  • 本地方法栈:类似“ JAVA虚拟机栈 ”,但是为native方法的运行提供内存环境。
  • JAVA堆:对象内存分配的地方,内存垃圾回收的主要区域,所有线程共享。可分为新生代,老生代。
  • 方法区:用于存储已经被JVM加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。Hotspot中的“永久代”。
  • 运行时常量池:方法区的一部分,存储常量信息,如各种字面量、符号引用等。
  • 直接内存:并不是JVM运行时数据区的一部分, 可直接访问的内存, 比如NIO会用到这部分。

按照JVM规范,除了程序计数器不会抛出OOM外,其他各个内存区域都可能会抛出OOM。

最常见的OOM情况有以下三种:

  • java.lang.OutOfMemoryError: Java heap space ------>java堆内存溢出,此种情况最常见,一般由于内存泄露或者堆的大小设置不当引起。对于内存泄露,需要通过内存监控软件查找程序中的泄露代码,而堆大小可以通过虚拟机参数-Xms,-Xmx等修改。
  • java.lang.OutOfMemoryError: PermGen space ------>java永久代溢出,即方法区溢出了,一般出现于大量Class或者jsp页面,或者采用cglib等反射机制的情况,因为上述情况会产生大量的Class信息存储于方法区。此种情况可以通过更改方法区的大小来解决,使用类似-XX:PermSize=64m -XX:MaxPermSize=256m的形式修改。另外,过多的常量尤其是字符串也会导致方法区溢出。
  • java.lang.StackOverflowError ------> 不会抛OOM error,但也是比较常见的Java内存溢出。JAVA虚拟机栈溢出,一般是由于程序中存在死循环或者深度递归调用造成的,栈大小设置太小也会出现此种溢出。可以通过虚拟机参数-Xss来设置栈的大小。


4)OOM分析--heapdump

要dump堆的内存镜像,可以采用如下两种方式:

设置JVM参数-XX:+HeapDumpOnOutOfMemoryError,设定当发生OOM时自动dump出堆信息。不过该方法需要JDK5以上版本。
使用JDK自带的jmap命令。"jmap -dump:format=b,file=heap.bin <pid>"   其中pid可以通过jps获取。
dump堆内存信息后,需要对dump出的文件进行分析,从而找到OOM的原因。常用的工具有:

mat: eclipse memory analyzer, 基于eclipse RCP的内存分析工具。详细信息参见:http://www.eclipse.org/mat/,推荐使用。   
jhat:JDK自带的java heap analyze tool,可以将堆中的对象以html的形式显示出来,包括对象的数量,大小等等,并支持对象查询语言OQL,分析相关的应用后,可以通过http://localhost:7000来访问分析结果。不推荐使用,因为在实际的排查过程中,一般是先在生产环境 dump出文件来,然后拉到自己的开发机器上分析,所以,不如采用高级的分析工具比如前面的mat来的高效。
这个链接:http://www.ibm.com/developerworks/cn/opensource/os-cn-ecl-ma/index.html中提供了一个采用mat分析的例子 。

注意:因为JVM规范没有对dump出的文件的格式进行定义,所以不同的虚拟机产生的dump文件并不是一样的。在分析时,需要针对不同的虚拟机的输出采用不同的分析工具(当然,有的工具可以兼容多个虚拟机的格式)。IBM HeapAnalyzer也是分析heap的一个常用的工具。

Java 堆溢出
GC Roots 到对象之间有可达路径来避免垃圾回收机智清理这些对象 就会在对象数量达到最大堆的绒容量 然后产生内存溢出异常
Eclipse 的 Debug 页面中设置虚拟机参数
代码:
-verbose:gc -Xms20M -Xmx20M Xmn10M -XX:+PrintGCDaetails -XX:SurvivorRatio=8
JAVA堆的大小设置最小值 -Xms 
JAVA堆的大小设置最大值 -Xmx

解决异常
首先通过内存映射分析工具 如 Eclipse Memory Analyzer 堆 dump出的异常堆转储进行快照解析确认内存中的对象是否是必要的  也就是先分清楚是 内存泄漏 Memory Leak 还是Memory Overflow 如果是内存泄漏 可通过工具进一步查看泄露的对象到GC Roots的引用链 就能找到泄露对象是怎么通过路径与 GC Roots 相关联导致垃圾收集器无法回收他们如果不存在泄露 就检查堆参数 -Xmx 与 -Xms 与机器物理内存对比是否还可以调大 从代码上检测 是否是某些对象的生命周期过长持有状态时间过长 尝试减少代码运行期间的内存消耗


虚拟机栈 和 本地方法栈 溢出
HotSpot 虚拟机不区分 虚拟机栈 和 本地方法栈 因此Xoss(设置本地方法栈内存大小)参数虽然存在但是是无效的
虚拟机栈 和 本地方法栈两种异常:
线程请求栈的深度大于虚拟机允许的最大深度 抛出 StackOverflowError
如果虚拟机在扩展栈是无法申请到足够的内存空间 抛出 Out Of Menmory Error

注:单个线程下 无论是栈帧太大还是虚拟机栈容量太小当内存无法分配的时候抛出的都是StackOverflowError


运行时常量池溢出
如果向运行时常亮去添加内容最简单的做法就是使用 String.intern() 这个 Narive 方法 该方法作用是如果池中包含一个等于此String对象的字符串 则返回代表池中这个字符串的 String 对象 否则将该 String 对象 的字符串添加到常量池中并返回此 String 对象的引用-XX:PermSize 方法区最小 -XX:MaxPermSize 方法区最大值如果运行时常量池 OOM 提示信息为 OutOfMemoryError : PermGen space 表示运行时常量池属于方法区的一部分 


方法区溢出
方法区是用于存放 Class信息的 如类名 访问修饰符 常量池 等当前主流AOP框架 (Spring Hibernate 等) 都会使用到 CFLib 这类字节码技术对类进行加强 增强的类越多 就越需要越大的方法区来保证 动态生成的Class 可以载入内存在动态生成大量 Class 的应用中 需要特别注意类的回收状况  注:除了使用GCLib字节码增强意外 常见的还有 JSP 或者动态产生JSP的应用文件(JSP在第一次运行时要编译为JAVA类)或是基于 OSGi(注:  OSGi(Open Service Gateway Initiative)技术是Java动态化模块化系统的一系列规范。OSGi一方面指维护OSGi规范的OSGI官方联盟,另一方面指的是该组织维护的基于Java语言的服务(业务)规范。简单来说,OSGi可以认为是Java平台的模块层。) 的应用即使是同一个类呗不同的加载器加载也会视为不同的类

本机直接内存溢出
DirectMemory 可通过-XX:MaxDirectMemorySize 指定 如果不指定则和JAVA堆的最大值一样使用DirectByteBuffer分配内存也会抛出内存溢出异常 但是他抛出异常的时候并没有向计算机申请分配内存 而是通过计算机内存得知是否可以分配于是抛出异常申请分配内存的方法是unsafe.allocateMemory();

================ End

免责声明:文章转载自《Java OOM 常见情况》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇vue-cli3.0和element-ui及axios的安装使用UITableView套UITableView数据刷新下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

VMware VCSA 6.7基本功能

一、VMware Tools基本特性 二、快照功能 三、Template技术介绍 1、用Template技术批量部署虚拟机,先创建自定义规范 2、部署虚拟机 四、创建虚拟机参数说明 五、虚拟机内存技术 博客地址:https://blog.51cto.com/delxu/288682 六、关于虚拟机CPU 七、vCenter的用户角色 八...

win7远程连接ubuntu,出现灰屏解决方法

  问题: win7远程虚拟机ubuntu 12.04出现灰色屏幕 打开windows自带的远程桌面连接。输入ubuntu虚拟机的IP地址 可以连接上,输入username和password 点击OK,出现灰色屏幕 解决方法: 1、在虚拟机上的Dash Home搜索Desktop Sharing,并进行如下设置。 若是没有找到,安装GNOME桌...

Linux的安装注意事项

Linux简介:   Linux是一套免费使用和自由传播的类Unix操作系统,是一个基于POSIX和UNIX的多用户、多任务、支持多线程和多CPU的操作系统。 Linux能运行主要的UNIX工具软件、应用程序和网络协议。它支持32位和64位硬件。Linux继承了Unix以网络为核心的设计思想,是一个性能稳定的多用户网络操作系统。       Linux的发...

manjaro:vmware workstation 16新建虚拟机提示:“Could not open /dev/vmmon: No such file or directory. Please make sure that the kernel module 'vmmon' is loaded.”解决

0x00 报错界面 0x02 解决办法   1、加载vmmon模块,命令: modprobe -a vmw_vmci vmmon   2、如果出现如下警告:   modprobe: WARNING: Module vmmon not found in directory /lib/modules/5.9.16-1-MANJARO 报错解决方案   1...

Vue实现双向绑定的原理以及响应式数据

一、vue中的响应式属性    Vue中的数据实现响应式绑定    1、对象实现响应式:      是在初始化的时候利用definePrototype的定义set和get过滤器,在进行组件模板编译时实现water的监听搜集依赖项,当数据发生变化时在set中通过调用dep.notify进行发布通知,实现视图的更新。   2、数组实现响应式:    对于数组则...

Visual Studio2015 简体中文版 安装

VS2015简体中文版安装 导航 介绍 解决安装先决条件 安装 VS2015 创建桌面快捷方式 启动 VS2015 命令启动VS2015 配置 VS2015 启动完成 Visual Studio的功能添加和删除 Visual Studio的卸载 MSDN介绍 MSDN安装 第一步 移动默认存储路径 第二步 剪切粘贴帮助文档 启动MSDN...