内存溢出原因与解决方案(4大主流方案详解)

简介: 本文详解内存溢出(OOM)的原因及解决方案。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。

关注△mikechen的互联网架构△,10年+BAT架构经验倾囊相授

大家好,我是 mikechen | 陈睿

内存溢出(OOM)是线上经常面临的问题,也是大厂面试经常考察的内容,下面我就全面来详解内存溢出(OOM)的原因及解决方案。@mikechen

内存溢出

内存溢出,简称OOM,内存溢出就是你要求分配的内存超出了系统能给你的,系统不能满足需求,于是产生溢出。

image.png

比如,申请了一个integer, 但给它存了long才能存下的数,那就是内存溢出。

再比如,你的电脑只有16GB的内存,已经把这16GB的内存用完了,但是还在继续用,就会造成内存溢出。

内存溢出的原因

程序常见的内存溢出原因:

  • 启动参数内存值设定的过小;
  • 内存中加载的数据量过于庞大;
  • 对象的引用使用完后未清空,使得 JVM 不能回收;
  • 代码中存在死循环或循环产生过多重复的对象实体。

Web服务器常见内存溢出提示

在不同的Web服务器或程序中,此错误常见的错误提示如下:

tomcat: java.lang.OutOfMemoryError: PermGen space
tomcat: java.lang.OutOfMemoryError: Java heap space
weblogic: Root cause of ServletException java.lang.OutOfMemoryError
resin: java.lang.OutOfMemoryError
java: java.lang.OutOfMemoryError

这些都是属于典型的内存溢出,接下来我们看看具体的内存溢出的解决办法。

内存溢出的解决办法

这里我就以Java举例,常见的Java内存溢出主要是方式:

image.png

堆溢出

这是很常见的一种内存溢出,报错信息如下:

java.lang.OutOfMemoryError: Java heap space
举一个死循环的例子,源码如下:

package com.mikechen.jvm;

/**

 * JVM 内存溢出案例详解
 *
 * @author  mikechen
 *
 */

import java.util.ArrayList;
import java.util.List;

public class MemoryOOM {

    static class Obj{ }

    /** * -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError * */

    public static void main(String[] args) {
        List<Obj> list = new ArrayList<>();
        try {
            while (true){
                list.add(new Obj());
            }
        }catch (Throwable t){
            t.printStackTrace();
            System.out.println("集合大小"+list.size());
        }
    }
}

这里会不断死循环list.add(new Obj()),不断的创建对象,当不总容量触及最大堆容量时,就会产生溢出,出现如下提示:

image.png

解决办法:

1、首先排查,检查是否存在大对象的分配,最有可能的是大对象分配。
2、其次,再通过jmap命令,把堆内存dump下来,使用mat工具分析一下,检查是否存在内存泄露的问题。
3、如果没有找到明显的内存泄露,可以适当调整-Xms和-Xmx两个JVM参数。

比如上面的例子,在第二个步骤通过dump下来的堆,用工具可以分析出死循环的位置,就可以及时解决堆溢出。

如果通过上面的两个步骤都排查后,没有发现问题,再调整JVM堆的初始化参数,问题基本就可以解决。

堆内存溢出

首先搞清楚java栈空间存储的是什么,栈内存可以分为虚拟机栈(VM Stack)和本地方法栈(Native Method Stack)。

每个方法执行时都会在java栈空间产生一个栈帧,存放方法的变量表,返回值等信息,方法的执行到结束就是一个栈帧入栈到出栈的过程。

具体来说,当线程执行某个方法时,JVM会创建栈帧并压栈,此时刚压栈的栈帧就成为了当前栈帧。

如果该方法进行递归调用时,JVM每次都会将保存了当前方法数据的栈帧压栈,每次栈帧中的数据都是对当前方法数据的一份拷贝,如果递归的次数足够多,多到栈中栈帧所使用的内存超出了栈内存的最大容量,此时JVM就会抛出StackOverflowError。

还是看一个例子:

public class stack{

public void test(){

    this.test();
}
public static void main(String[] args){
    for(; ; ;)
        new stack().test;
        }
}

最后就会出现:

StackOverflowError的错误提示

总之,不论是因为栈帧太大还是栈内存太小,当新的栈帧内存无法被分配时,JVM就会抛出StackOverFlowError,通常栈内存可以通过设置-Xss参数来改变大小。

永久代/元空间溢出
永久区(Perm) 存放了被虚拟机加载的类信息、常量、静态变量、JIT编译后的代码等。

在JDK 1.8中,永久区被一块称为元数据的区域替代, 但是它们的功能是类似的,都是为了保存类的元信息,如果一 一个系统定了太多的类型,那么永久区是有可能溢出的,报错信息如下:

java.lang.OutOfMemoryError: PermGen spacejava.lang.OutOfMemoryError: Metaspace
解决方法有如下几种:

1、检查是否永久代空间或者元空间设置的过小

2、检查代码中是否存在大量的反射操作

3、dump之后通过mat检查是否存在大量由于反射生成的代理类

4、放大招,重启JVM

以上,是内存溢出原因与解决方案的详细解析,欢迎评论区留言交流或拓展。

我是 mikechen | 陈睿 ,关注【mikechen的互联网架构】,10年+BAT架构技术倾囊相授。

新的架构专题内容,第一时间更新至:阿里架构师进阶全部合集

本文已同步我的技术博客 www.mikechen.cc,更新至我原创的《30W+字阿里架构技术合集》中。

相关文章
|
缓存 运维 Shell
幻兽帕鲁爆内存优化方案
最近有很多小伙伴反馈说4C16G的服务器玩时间久了经常出现内存过满自动重启的情况,现在总结下linux和windows系统下怎么进行优化。
|
运维 监控 Java
内存溢出+CPU占用过高:问题排查+解决方案+复盘(超详细分析教程)
全网最全的内存溢出CPU占用过高排查文章,包含:问题出现现象+临时解决方案+复现问题+定位问题发生原因+优化代码+优化后进行压测,上线+复盘
3601 5
|
4月前
|
Web App开发 缓存 监控
内存溢出与内存泄漏:解析与解决方案
本文深入解析内存溢出与内存泄漏的区别及成因,结合Java代码示例展示典型问题场景,剖析静态集合滥用、资源未释放等常见原因,并提供使用分析工具、优化内存配置、分批处理数据等实用解决方案,助力提升程序稳定性与性能。
1168 1
|
8月前
|
缓存 监控 Cloud Native
Java Solon v3.2.0 高并发与低内存实战指南之解决方案优化
本文深入解析了Java Solon v3.2.0框架的实战应用,聚焦高并发与低内存消耗场景。通过响应式编程、云原生支持、内存优化等特性,结合API网关、数据库操作及分布式缓存实例,展示其在秒杀系统中的性能优势。文章还提供了Docker部署、监控方案及实际效果数据,助力开发者构建高效稳定的应用系统。代码示例详尽,适合希望提升系统性能的Java开发者参考。
432 4
Java Solon v3.2.0 高并发与低内存实战指南之解决方案优化
|
12月前
|
监控 Java 计算机视觉
Python图像处理中的内存泄漏问题:原因、检测与解决方案
在Python图像处理中,内存泄漏是常见问题,尤其在处理大图像时。本文探讨了内存泄漏的原因(如大图像数据、循环引用、外部库使用等),并介绍了检测工具(如memory_profiler、objgraph、tracemalloc)和解决方法(如显式释放资源、避免循环引用、选择良好内存管理的库)。通过具体代码示例,帮助开发者有效应对内存泄漏挑战。
602 1
|
运维 监控 Java
为何内存不够用?微服务改造启动多个Spring Boot的陷阱与解决方案
本文记录并复盘了生产环境中Spring Boot应用内存占用过高的问题及解决过程。系统上线初期运行正常,但随着业务量上升,多个Spring Boot应用共占用了64G内存中的大部分,导致应用假死。通过jps和jmap工具排查发现,原因是运维人员未设置JVM参数,导致默认配置下每个应用占用近12G内存。最终通过调整JVM参数、优化堆内存大小等措施解决了问题。建议在生产环境中合理设置JVM参数,避免资源浪费和性能问题。
1064 3
|
Web App开发 缓存 JavaScript
技术分享:深入探索内存泄漏——识别、分类与解决方案
【8月更文挑战第27天】在软件开发的浩瀚星海中,内存管理始终是程序员们必须面对的重要课题。内存泄漏,作为内存管理不善的典型症状,不仅影响应用性能,还可能导致系统崩溃,是每位开发者都需警惕的“暗礁”。本文将带您深入探索内存泄漏的本质、常见类型及有效的解决策略,助力您的工作学习之旅更加顺畅。
257 0
|
存储 缓存 NoSQL
Redis是一种高性能的内存数据库,常用于高并发环境下的缓存解决方案
【6月更文挑战第18天】**Redis摘要:** 高性能内存数据库,擅长高并发缓存。数据存内存,访问迅速;支持字符串、列表等多元数据类型;具备持久化防止数据丢失;丰富命令集便于操作;通过节点集群实现数据分片与负载均衡,增强可用性和扩展性。理想的缓存解决方案。
326 1
Python成员属性的内存特性与底层内存优化方案
这篇博客主要分享一下python成员属性的内存特性,也就是python底层节约内存的优化方案
|
设计模式 安全 Java
Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
327 0