Java内存泄漏概念、造成原因及检测方式(全)

简介: 本身java有垃圾回收器GC,可以内存管理,但为什么还会造成内存泄漏(内存泄漏不等于内存溢出),内存泄漏在项目实战或者企业项目是不被允许,甚至在企业面试中也是常考的题型。

前言

本身java有垃圾回收器GC,可以内存管理,但为什么还会造成内存泄漏(内存泄漏不等于内存溢出),内存泄漏在项目实战或者企业项目是不被允许,甚至在企业面试中也是常考的题型

1. 概念

了解什么是内存泄漏,需要知道具体的定义、检测以及解决方式

内存泄漏:对系统申请内存使用,将其内存分配给对象使用,但内存空间使用完毕后未释放,一直占用内存空间。(长期的堆积,内存迟早被耗尽,所以今早的解决内存泄漏无疑是好事)

本身内存泄漏就是缺点,没有所谓的优点
缺点如下

  • 造成OOM
  • 性能整体下降导致一系列错误

(系统分配额外内存影响整体系统的运行情况)

  • 程序运行延迟卡顿或者直接奔溃

(可用内存少,频繁GC,频率高了,用户会感受到卡顿)

2. 原因

以下的方式会造成内存泄漏的风险,所以谨慎使用

2.1 大量使用static静态变量

回顾下static的知识点:java零基础从入门到精通(全)

  • 实例变量是对象级别的,每个对象的实例变量值可能不同,所以实例变量必须先创建对象,通过“引用”去访问,而静态变量访问时不需要创建对象,直接通过“类名”访问。实例变量存储在堆内存当中,静态变量存储在方法区当中。实例变量在构造方法执行过程中初始化,静态变量在类加载时初始化
  • 当一个类的所有对象的某个“属性值”不会随着对象的改变而变化的时候,建议将该属性定义为静态属性(或者说把这个变量定义为静态变量),静态变量在类加载的时候初始化,存储在方法区当中,不需要创建对象,直接通过“类名”来访问

类似的伪代码如下:

public class test{
    // 静态变量
    public void method() {
        // 该函数中使用了静态变量
    }
    public static void main(String[] args) {
        // 业务逻辑代码
        
        // 此处调用了method来操作静态变量
        
        // 业务逻辑代码
    }
}

如上就会造成一时的内存泄漏,直到程序运行完毕才会释放
这是因为静态变量的生命周期和主函数保持一致,所以尽量减少使用静态变量

2.2 finalize方法

(补充一下常考的面试题)
这个方法常和final、finally与finalize作比较

  • final是一个关键字。表示最终的。不变的
  • finally也是一个关键字,和try联合使用,使用在异常处理机制中
  • finalize()是Object类中的一个方法。作为方法名出现,finalize是标识符。这个方法是由垃圾回收器GC负责调用的

以下是finalize的整体流程:
在这里插入图片描述
之所以会造成内存泄漏,是因为在垃圾回收的时候,如果重写了finalize方法而且该对象的finalize方法没有被执行过,但是进入队列之后一直没被调用就会一直占用内存空间

2.3 对象引用有误

业务逻辑代码模块中,稍微不注意就可能引发内存泄漏,一句代码也可能引发致命错误

具体如下:

int[] a = new int[5];
int[] b = new int[5];
b = a;

原本GC清除的是引用地址,此处这样调用,b的引用地址就会变成了a,导致b的引用地址就会找不到,GC内存回收的时候就会造成内存泄漏。(尽量避免这种情况的发生)

2.4 资源未被关闭

在使用到jdbc、数据库连接、io连接等,都会进行一个jvm的分配内存
最后代码模块都会进行一个资源的关闭,如果资源未被正常关闭,内存就一直在运行,直到主函数关闭(但是并不是时时刻刻都在连接查询资源)

补充:对应的知识可看我之前的文章:

  1. javaSE从入门到精通的二十万字总结(二)
  2. jdbc从入门到精通(全)
  3. Mysql底层原理详细剖析+常见面试题(全)

2.5 Threadlocal对象赋值null

补充以下ThreadLocal的底层原理以及知识点:
每个线程都有自个独立的ThreadLocalMap对象(Entry对象)
如果当前线程对应的ThreadLocalMap对象为空,则为其创建key以及value(key为ThreadLocal对象,value为缓存变量值)

Threadlocal<String> stringThreadlocal=new Threadlocal<>();
stringThreadlocal.set("码农研究僧"); 

ThreadLocalMap可以存放多个ThreadLocal对象
每个ThreadLocal对象只能缓存一个变量值

通过ThreadLocal.get()可以获取相对应的key值
ThreadLocalMap 的entry对象为,key为ThreadLocal,value为缓存变量值


主线程调用了ThreadLocal,给予其变量为null(引用断开了,而不是单纯没有引用)。但是当前线程的ThreadLocalMap还是会引用其堆的变量,如果是强引用,gc是不会回收的。
代码如下所示:(该变量不会被gc回收)

ThreadLocal<String>ss =new ThreadLocal<>();
ss.set("码农研究僧");
ss=null

如果修改其变量引用指向,才会被gc回收

ThreadLocalMap删除Entry的对象,才会解决其引用断开,所以才会被gc清理掉
代码如下所示:

ThreadLocal<String>ss =new ThreadLocal<>();
ss.set("码农研究僧");
//ThreadLocalMap与堆内存中的ThreadLocal断开引用
ss.remove();
//ss与堆内存中的ThreadLocal断开引用
ss=null

GCroot引用链就会发现ThreadLocal没有被任何人引用就会被清理掉该对象,避免内存泄漏问题

关于避免ThreadLocal内存泄漏问题
可以通过:

  1. 通过调用remove方法将不要的数据移除避免内存泄漏问题
  2. 每次set设置的时候(内部方法都会判断之前的key是否为null),如果设置了null,则可以避免内存泄漏

2.6 其他

如果在实际应用场景中遇到一些内存泄漏的问题
可在底下评论区留言

3. 检测

3.1 JVM命令

如果不使用工具来检测的话,可通过JVM自带的一些命令参数:

  • jps:当前运行的所有java进程
  • jstat:单个java进程GC情况
  • jmap: 单个java进程中堆内存使用情况
  • jvisualvm:可视化查看堆内存与metaspace占用情况
  • jstack:查看具体某个java进行的线程堆栈情况

也可通过IDEA配置-verbose:gc

3.2 工具

JVM的检测巩固可通过java的VisualVM、YourKit、Netbeans Profiler等

  • JProbe:分析java内存泄漏
  • Jprofiler:主要用于分析J2SE和J2EE的一些应用程序
  • JRockit:诊断java内存泄漏找出根本原因

通过分析内存、对象以及CPU的各个情况
查看堆的最大最小以及使用情况

通过查看堆内存,打印堆的各个使用情况
或者将其Dump文件分析

对应的检测工具跟命令大同小异,都是分析其对应内存是否有泄漏情况

相关文章
|
27天前
|
存储 缓存 安全
Java内存模型深度解析:从理论到实践####
【10月更文挑战第21天】 本文深入探讨了Java内存模型(JMM)的核心概念与底层机制,通过剖析其设计原理、内存可见性问题及其解决方案,结合具体代码示例,帮助读者构建对JMM的全面理解。不同于传统的摘要概述,我们将直接以故事化手法引入,让读者在轻松的情境中领略JMM的精髓。 ####
33 6
|
18天前
|
安全 Java 程序员
深入理解Java内存模型与并发编程####
本文旨在探讨Java内存模型(JMM)的复杂性及其对并发编程的影响,不同于传统的摘要形式,本文将以一个实际案例为引子,逐步揭示JMM的核心概念,包括原子性、可见性、有序性,以及这些特性在多线程环境下的具体表现。通过对比分析不同并发工具类的应用,如synchronized、volatile关键字、Lock接口及其实现等,本文将展示如何在实践中有效利用JMM来设计高效且安全的并发程序。最后,还将简要介绍Java 8及更高版本中引入的新特性,如StampedLock,以及它们如何进一步优化多线程编程模型。 ####
21 0
|
20天前
|
存储 监控 算法
Java内存管理深度剖析:从垃圾收集到内存泄漏的全面指南####
本文深入探讨了Java虚拟机(JVM)中的内存管理机制,特别是垃圾收集(GC)的工作原理及其调优策略。不同于传统的摘要概述,本文将通过实际案例分析,揭示内存泄漏的根源与预防措施,为开发者提供实战中的优化建议,旨在帮助读者构建高效、稳定的Java应用。 ####
32 8
|
18天前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
22天前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
48 5
|
20天前
|
存储 算法 Java
Java内存管理深度解析####
本文深入探讨了Java虚拟机(JVM)中的内存分配与垃圾回收机制,揭示了其高效管理内存的奥秘。文章首先概述了JVM内存模型,随后详细阐述了堆、栈、方法区等关键区域的作用及管理策略。在垃圾回收部分,重点介绍了标记-清除、复制算法、标记-整理等多种回收算法的工作原理及其适用场景,并通过实际案例分析了不同GC策略对应用性能的影响。对于开发者而言,理解这些原理有助于编写出更加高效、稳定的Java应用程序。 ####
|
20天前
|
安全 Java 程序员
Java内存模型的深入理解与实践
本文旨在深入探讨Java内存模型(JMM)的核心概念,包括原子性、可见性和有序性,并通过实例代码分析这些特性在实际编程中的应用。我们将从理论到实践,逐步揭示JMM在多线程编程中的重要性和复杂性,帮助读者构建更加健壮的并发程序。
|
25天前
|
算法 Java 开发者
Java内存管理与垃圾回收机制深度剖析####
本文深入探讨了Java虚拟机(JVM)的内存管理机制,特别是其垃圾回收机制的工作原理、算法及实践优化策略。不同于传统的摘要概述,本文将以一个虚拟的“城市环卫系统”为比喻,生动形象地揭示Java内存管理的奥秘,旨在帮助开发者更好地理解并调优Java应用的性能。 ####
|
26天前
|
Java
java内存区域
1)栈内存:保存所有的对象名称 2)堆内存:保存每个对象的具体属性 3)全局数据区:保存static类型的属性 4)全局代码区:保存所有的方法定义
22 1
|
17天前
|
存储 监控 算法
Java内存管理的艺术:深入理解垃圾回收机制####
本文将引领读者探索Java虚拟机(JVM)中垃圾回收的奥秘,解析其背后的算法原理,通过实例揭示调优策略,旨在提升Java开发者对内存管理能力的认知,优化应用程序性能。 ####
32 0