OOM和StackOverFlow的区别

1、StackOverFlowError
Thrown when a stack overflow occurs because an application recurses too deeply.

调用栈深度超过限制

递归运算时会遇到

2、OOM
Out of memory
当JVM分配内存时 不够才会抛出异常
主要分为

java.lang.OutOfMemoryError:Java heap space
新建对象时 JVM内存不足 new

java.lang.OutOfMemoryError:GC overhead limit exceedec
GC回收时间过长时会抛出该异常
过长的定义是,超过98%的时间用来做GC并且回收了不到2%的堆内存
连续多次GC都只回收了不到2%的极端情况下才会抛出。
避免恶性循环 浪费CPU性能

java.lang.OutOfMemoryError:Direct buffer memory
写NIO程序时经常使用ByteBuffer来读取或者写入数据,这是一种基于通道(channel)与缓冲区(buffer)的I/O方式,他可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆里面的DirectByteBuffer对象作为这块内存的引用操作,这样能在一些场景中显著提高性能,因为避免了在Java堆和Native堆中来回复制数据
ByteBuffer.allocate(caoability)第一种方式是分配JVM堆内存,属于GC管辖范围,由于需要拷贝所以速度相对较慢
ByteBuffer.allocateDirect(caoability) 第2种方式是分配OS本地内存,不属于GC管辖范围,由于不需要内存拷贝所以速度相对较快。
但如果不断分配本地内存,堆内存很少使用,那么JVM就不需要执行GC,DirectByteBuffer对象们就不会被回收,这时候堆内存充足,但本地内存可能已经使用光了,再次尝试分配本地内存就会出现OOM

java.lang.OutOfMemoryError:unable to create new native thread
导致原因:
1.你的应用创建了太多线程,一个应用进程创建多个线程,超过系统承载极限
2、你的服务器不允许你的应用程序创建这么多线程,linux系统默认允许单个进程可以创建的线程数是1024个

java.lang.OutOfMemoryError:Metaspce
Java 8 之后的版本使用metaspace 来替代永久代
metaspace 是方法区在HotSpot中的实现,它与永久代最大的区别在于:metaspace并不在虚拟机内存中而是使用本地内存
存储信息如下:
虚拟机加载的类信息
常量池
静态变量
即时编译后的代码

Enhancer

发布了328 篇原创文章 · 获赞 23 · 访问量 7万+

猜你喜欢

转载自blog.csdn.net/lbh199466/article/details/104498777
OOM