当前位置: 首页 > news >正文

OOM不会导致JVM退出

问题来源

一次生产事故,由于一次性从数据库查询过多数据导致线程 OOM:Java heap space 异常(千万级表,JVM堆内存2G),但是在线程OOM发生时,java进程却没有立即挂掉。

##OOM与异常
说到底OutOfMemoryError也只是一个java中的异常而已,属于Error一系非检查异常:

ObjectThrowableErrorVirtualMachineErrorOutOfMemoryError

堆内存不够与异常的关系

线程发生OOM Java heap space,首先是堆空间不够了,然后再由jvm在申请分配空间的方法调用上抛出OOM异常。
对于线程,它会像处理普通异常一样,处理OutOfMemoryError。

实例

package org.example;import com.sun.javafx.scene.control.skin.TableHeaderRow;import java.util.ArrayList;//TIP To <b>Run</b> code, press <shortcut actionId="Run"/> or
// click the <icon src="AllIcons.Actions.Execute"/> icon in the gutter.
public class Main {public static void main(String[] args) throws InterruptedException{//TIP Press <shortcut actionId="ShowIntentionActions"/> with your caret at the highlighted text// to see how IntelliJ IDEA suggests fixing it.System.out.printf("Hello and welcome!");System.out.println("JVM从OS获取的 最大 内存" + Runtime.getRuntime().maxMemory() / 1024 / 1024 + "M");System.out.println("JVM从OS获取的 当前 内存" + Runtime.getRuntime().totalMemory() / 1024 / 1024 + "M");System.out.println("JVM从OS获取的 但空闲 内存" + Runtime.getRuntime().freeMemory() / 1024 / 1024 + "M");// 创建线程new Thread(() -> {ArrayList<byte[]> bytes = new ArrayList<>();for (int i = 0; i < 2000; i++) {byte[] bytes1 = new byte[3 * 1024 * 1024];bytes.add(bytes1);System.out.println(Thread.currentThread().getName()+":JVM从OS获取的 当前 内存" + Runtime.getRuntime().totalMemory() / 1024 / 1024 + "M");}}).start();// 创建线程new Thread(() -> {ArrayList<byte[]> bytes = new ArrayList<>();for (int i = 0; i < 2000; i++) {byte[] bytes1 = new byte[3 * 1024 * 1024];bytes.add(bytes1);System.out.println(Thread.currentThread().getName()+":JVM从OS获取的 当前 内存" + Runtime.getRuntime().totalMemory() / 1024 / 1024 + "M");}}).start();while (true){System.out.println(Thread.currentThread().getName() + " continuing...");Thread.sleep(1000L);}}
}

结果:

##本机64位,16G内存,默认最大堆3600M
JVM从OS获取的 最大 内存3600M
JVM从OS获取的 当前 内存243M
JVM从OS获取的 但空闲 内存237M
main continuing...##2个线程分别开始创建byte数组,占用内存 
Thread-0:JVM从OS获取的 当前 内存307M
Thread-1:JVM从OS获取的 当前 内存307M
Thread-1:JVM从OS获取的 当前 内存307M
Thread-0:JVM从OS获取的 当前 内存307M##JVM增长到极限,Thread-1先报OOM
Thread-0:JVM从OS获取的 当前 内存3366M
Thread-0:JVM从OS获取的 当前 内存3366M
Exception in thread "Thread-1" java.lang.OutOfMemoryError: Java heap spaceat org.example.Main.lambda$main$1(Main.java:39)at org.example.Main$$Lambda$2/1828972342.run(Unknown Source)at java.lang.Thread.run(Thread.java:750)
Thread-0:JVM从OS获取的 当前 内存3366M
Thread-0:JVM从OS获取的 当前 内存3366M#Thread-1失败终止后,起引用的对象也就可以GC了,Thread-0又获取了好多内存,直到OOM
Thread-0:JVM从OS获取的 当前 内存3616M
Thread-0:JVM从OS获取的 当前 内存3616M
Exception in thread "Thread-0" java.lang.OutOfMemoryError: Java heap spaceat org.example.Main.lambda$main$0(Main.java:29)at org.example.Main$$Lambda$1/1989780873.run(Unknown Source)at java.lang.Thread.run(Thread.java:750)
main continuing...
main continuing...
main continuing...

结论

1、OOM是在线程上发生的,会被当做一般异常处理,不会导致JVM的退出;
2、多线程公用JVM,一个线程终止GC后,内存重新给其他线程分配。

http://www.lryc.cn/news/356074.html

相关文章:

  • C++学习日记 | LAB 6 static library 静态库
  • 他用AI,抄袭了我的AI作品
  • 力扣刷题--2956. 找到两个数组中的公共元素【简单】
  • 海信集团携纷享销客启动LTC数字化落地 推动ToB业务再升级
  • 【Go语言入门学习笔记】Part5.函数
  • 磁珠笔记汇总
  • 【css3】02-css3新特性之选择器篇
  • 修正错误的插入排序
  • Unity 权限 之 Android 【权限 动态申请】功能的简单封装
  • 跟进2年弄丢1.8亿,你的大客管理错在哪里?
  • 浅说线性DP(上)
  • leetcode题目18
  • 后端企业级开发之yaml数据序列化格式文件详解2024
  • 智能界面设计:数字孪生与大数据结合的美学典范
  • 听说部门来了个00后测试开发,一顿操作给我整麻了
  • Linux shell命令
  • 【Linux】Linux基本指令1
  • 重学java 49 增强for
  • BUUCTF靶场[Web] [极客大挑战 2019]Havefun1、[HCTF 2018]WarmUp1、[ACTF2020 新生赛]Include
  • 现代信号处理11_Spectral Analysis谱分析(CSDN_20240526)
  • C#开发上位机应用:基础与实践
  • 话术巧妙分隔沟通效果更佳看看这个小技巧
  • 【Spring】设计模式(GOF)
  • php抖音详情和关键词搜索api
  • SOCKS 代理 和 HTTP 代理
  • 【Linux】自己实现一个bash进程
  • 记录深度学习GPU配置,下载CUDA与cuDnn
  • Word将表格调成合适的大小
  • 2024HBCPC:C Goose Goose Duck
  • Llama 3 模型家族构建安全可信赖企业级AI应用之使用 Llama Guard 保护大模型对话 (八)