组织架构

严格来说,StackOverflowError和OutOfMemoryError都属于错误,而不是异常。

java.lang.StackOverflowError

 public class StackOverflowErrorDemo {
public static void method(){
method();
}
public static void main(String[] args) {
method();
}
}

在上例中,方法的深度调用,导致栈溢出。

java.lang.OutOfMemoryError

java.lang.OutOfMemoryError:Java heap space

 public class JavaHeapSpaceDemo {
private static List<String> list = new ArrayList<>();
public static void main(String[] args) {
String str = "";
while(true){
str += str + new Random().nextInt(111111);
list.add(str);
}
}
}

Java堆用于存储对象实例,不断创建对象,保证这些对象到GC Roots有可达路径,可以避免对象被垃圾回收,很快对象数量就会达到最大堆的容量限制,产生内存溢出异常。

java.lang.OutOfMemoryError:GC overhead limit exceeded

jdk1.6新增的错误类型,GC回收时间过长时会抛出OOM。超过98%的时间用来做GC,但是只回收了2%的堆内存,连续多次垃圾回收,只回收了不到2%的极端情况才会抛出。

经过垃圾回收释放的2%可用内存空间会快速的被填满,迫使GC再次执行,出现频繁的执行GC操作, 服务器会因为频繁的执行GC垃圾回收操作而达到100%的使用率,服务器运行变慢,应用系统会出现卡死现象,平常只需几毫秒就可以执行的操作,现在需要更长时间,甚至是好几分钟才可以完成。

如果不抛出GC overhead limit exceeded,GC会频繁的执行,但是被占用的内存,经过多次长时间的GC操作都无法回收,导致可用内存越来越少,俗称内存泄露。

 public class GCOverheadDemo {
public static void main(String[] args) {
int i =0;
List<String> list = new ArrayList<>();
while (true){
list.add(String.valueOf(++i).intern());
}
}
}

String.intern()是一个Native方法,底层调用C++的 StringTable::intern方法实现。

当通过语句str.intern()调用intern()方法后,JVM 就会在当前类的常量池中查找是否存在与str等值的String,若存在则直接返回常量池中相应Strnig的引用;若不存在,则会在常量池中创建一个等值的String,然后返回这个String在常量池中的引用。

在此为了快速产生这种异常,首先配置VM options为【-Xms10m -Xmx10m -XX:+PrintGCDetails -XX:MaxDirectMemorySize=5m】,再通过intern()循环创建字符串。从GC的打印日志可以看出,虽然一直在JVM执行GC,但是回收没有效果,最终抛出异常。

java.lang.OutOfMemoryError:Direct buffer memory

产生这种异常的原因是,写NIO程序经常使用ByteBuffer来读取或写入,NIO是一种基于通道和缓冲区的IO方式,它可以使用本地函数库直接分配堆外内存,通过存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作,这样可以在一些情况下显著提升性能,它避免了在Java堆和Native堆中来回复制数据。

ByteBuffer有两种分配内存的方式:

  1. allocate:分配JVM堆内存,属于GC范畴,需要拷贝数据,速度慢。
  2. allocateDirect:分配OS本地内存,不属于GC范畴,不需要内存拷贝,速度较快。

如果不断分配本地内存,堆内存很少使用,JVM也就不需要GC,DirectByteBuffer对象就不会被回收,如果本地内存快被用完了,之后再分配本地内存就会抛出OOM。

java.lang.OutOfMemoryError:unable to create new native thread

高并发请求服务器的时候,经常会出现该异常,该异常与平台有关,导致的原因是应用创建了太多线程,一个应用进程创建多个线程,超过系统承载的极限,服务器不允许创建这么多线程,linux默认允许单个进程创建的线程数是1024个。如果超过允许的值,就会抛出异常。

解决的方法是减少创建线程的数量,尽可能少的创建线程。或者修改服务器配置,扩大限制的线程数量。

java.lang.OutOfMemoryError:Metaspace

Java8之后元空间取代了永久代,元空间的本质和永久代类似,都是JVM规范中方法区的实现,其区别在于元空间不虚拟机中,而是在本地内存中,默认情况下,元空间大小仅受内存限制。主要存放:虚拟机加载的类信息、常量池、静态变量、即时编译后的代码。

解决的方法是通过配置-XX:MaxMetaspaceSize=512m参数,增大Metaspace的空间。

还可以直接去掉 Metaspace 的大小限制。 但是,如果不限制Metaspace内存的大小,当物理内存过载的时候,有可能会引起内存交换,严重拖累系统性能。此外,还可能造成native内存分配失败等问题。

【JVM】关于OOM的二三事的更多相关文章

  1. Java并发编程二三事

    Java并发编程二三事 转自我的Github 近日重新翻了一下<Java Concurrency in Practice>故以此文记之. 我觉得Java的并发可以从下面三个点去理解: * ...

  2. linux杂记(十二?) 关于账号和密码的二三事

    关于密码的二三事 关于账号和密码的二三事 久了不更linux的相关知识,实在是懒得想内容点(纯粹是懒).那么今天就来谈谈关于linux密码和账号的重要概念. 假如你的主机遭到入侵,那么对方的第一个侵入 ...

  3. MySQL5.7关于密码二三事

    MySQL5.7关于密码二三事 第一个:update user set password=password('root') where user='root' and host='localhost' ...

  4. Java中的匿名内部类及内部类的二三事

    匿名内部类适合创建那些只需要使用一次的类,它的语法有些奇怪,创建匿名内部类会立即创建一个该类的实例,这个类定义立即消失,且不能重复使用. 定义匿名类的格式如下: new 实现接口() |父类构造器(实 ...

  5. JVM 字节码(二)方法表详解

    JVM 字节码(二)方法表和属性表 上一节中对 ClassFile 的整体进行了五个详细的说明, 本节围绕 ClassFile 最重要的一个内容 - 方法表的 Code 属性展开 ,更多 JVM Me ...

  6. JVM 内部原理(二)— 基本概念之字节码

    JVM 内部原理(二)- 基本概念之字节码 介绍 版本:Java SE 7 每位使用 Java 的程序员都知道 Java 字节码在 Java 运行时(JRE - Java Runtime Enviro ...

  7. Spark On Yarn Cluster生产环境下JVM的OOM和Stack Overflow问题

    1.Spark on Yarn下JVM的OOM问题及解决方式 2.Spark中Driver的Stack Overflow的问题及解决方式 Spark on Yarn cluster mode: 此时有 ...

  8. Emacs 启动优化二三事

    Emacs 启动优化二三事 */--> div.org-src-container { font-size: 85%; font-family: monospace; } p {font-siz ...

  9. WinForm二三事(三)Control.Invoke&Control.BeginInvoke

    http://www.cnblogs.com/yuyijq/archive/2010/01/11/1643802.html 这个系列从2009年写到2010年,差点又成太监文.随着WPF/Silver ...

随机推荐

  1. Mysql数据库多表联查

    内连接:查询的是多个表的交集 外连接:查询的是一张表的全部数据和另一张表满足要求的数据 student数据库表 grade数据库表 内连接: 隐式内连接 SELECT s.id, s. NAME, s ...

  2. Spring官网阅读(十四)Spring中的BeanWrapper及类型转换

    文章目录 接口定义 继承关系 接口功能 1.PropertyEditorRegistry(属性编辑器注册器) 接口定义 PropertyEditor 概念 Spring中对PropertyEditor ...

  3. 201771010113 李婷华 《面向对象程序设计(Java)》第十六周总结

    一.理论知识部分 1.程序是一段静态的代码,它应用程序执行蓝 是一段静态的代码,它应用程序执行蓝 是一段静态的代码,它应用程序执行蓝本. 2.进程是程序的一次动态执行,它对应了从代码加载.执行至执行完 ...

  4. OpenWrt R2020.3.19 反追踪 抗污染 加速 PSW 无缝集成 UnPnP NAS

    固件说明 基于Lede OpenWrt R2020.3.19版本Lienol Feed及若干自行维护的软件包 结合家庭x86软路由场景需要定制 按照家庭应用场景对固件及软件进行测试,通过后发布 设计目 ...

  5. vue滑动吸顶以及锚点定位

    Vue项目中需要实现滑动吸顶以及锚点定位功能.template代码如下: <template> <div class="main"> <div id= ...

  6. C# 数据操作系列 - 3. ADO.NET 离线查询

    0. 前言 在上一篇中,我故意留下了查询的示范没讲.虽然说可以通过以下代码获取一个DataReader: IDataReader reader = command.ExecuteReader(); 然 ...

  7. elasticsearch kibana + 分词器安装详细步骤

    elasticsearch kibana + 分词器安装详细步骤 一.准备环境 系统:Centos7 JDK安装包:jdk-8u191-linux-x64.tar.gz ES安装包:elasticse ...

  8. CSS之未知高度img垂直居中

    测试代码如下:(能够水平居中,通过text-align:center实现) <style>.box{ width:800px;height:600px;border:2px solid # ...

  9. Python-实现sign签名接口校

    前言 在之前的随笔中,我们已经学过了如何使用使用JMeter和Postman实现sign签名接口校验的接口测试,今天我们来学习一下如何写Python脚本实现签名接口的接口测试. 签名接口: 地址: h ...

  10. wepy+vant-weapp踩坑记

    最近用了几个月的wepy框架,碰到了挺多问题,这里总结一下 1.clone的代码无法再本地运行,wepy报错 解决方案: 执行命令 : `npm install wepy-cli@1.6.1-alph ...