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文件分析

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

相关文章
|
6天前
|
安全 Java 程序员
深入理解Java内存模型与并发编程####
本文旨在探讨Java内存模型(JMM)的复杂性及其对并发编程的影响,不同于传统的摘要形式,本文将以一个实际案例为引子,逐步揭示JMM的核心概念,包括原子性、可见性、有序性,以及这些特性在多线程环境下的具体表现。通过对比分析不同并发工具类的应用,如synchronized、volatile关键字、Lock接口及其实现等,本文将展示如何在实践中有效利用JMM来设计高效且安全的并发程序。最后,还将简要介绍Java 8及更高版本中引入的新特性,如StampedLock,以及它们如何进一步优化多线程编程模型。 ####
14 0
|
1月前
|
存储 Java 编译器
Java内存模型(JMM)深度解析####
本文深入探讨了Java内存模型(JMM)的工作原理,旨在帮助开发者理解多线程环境下并发编程的挑战与解决方案。通过剖析JVM如何管理线程间的数据可见性、原子性和有序性问题,本文将揭示synchronized关键字背后的机制,并介绍volatile关键字和final关键字在保证变量同步与不可变性方面的作用。同时,文章还将讨论现代Java并发工具类如java.util.concurrent包中的核心组件,以及它们如何简化高效并发程序的设计。无论你是初学者还是有经验的开发者,本文都将为你提供宝贵的见解,助你在Java并发编程领域更进一步。 ####
|
8天前
|
存储 监控 算法
Java内存管理深度剖析:从垃圾收集到内存泄漏的全面指南####
本文深入探讨了Java虚拟机(JVM)中的内存管理机制,特别是垃圾收集(GC)的工作原理及其调优策略。不同于传统的摘要概述,本文将通过实际案例分析,揭示内存泄漏的根源与预防措施,为开发者提供实战中的优化建议,旨在帮助读者构建高效、稳定的Java应用。 ####
22 8
|
14天前
|
Java
java内存区域
1)栈内存:保存所有的对象名称 2)堆内存:保存每个对象的具体属性 3)全局数据区:保存static类型的属性 4)全局代码区:保存所有的方法定义
20 1
|
28天前
|
缓存 算法 Java
本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制
在现代软件开发中,性能优化至关重要。本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制。通过调整垃圾回收器参数、优化堆大小与布局、使用对象池和缓存技术,开发者可显著提升应用性能和稳定性。
45 6
|
5天前
|
存储 监控 算法
Java内存管理的艺术:深入理解垃圾回收机制####
本文将引领读者探索Java虚拟机(JVM)中垃圾回收的奥秘,解析其背后的算法原理,通过实例揭示调优策略,旨在提升Java开发者对内存管理能力的认知,优化应用程序性能。 ####
19 0
|
1月前
|
存储 缓存 安全
Java内存模型(JMM):深入理解并发编程的基石####
【10月更文挑战第29天】 本文作为一篇技术性文章,旨在深入探讨Java内存模型(JMM)的核心概念、工作原理及其在并发编程中的应用。我们将从JMM的基本定义出发,逐步剖析其如何通过happens-before原则、volatile关键字、synchronized关键字等机制,解决多线程环境下的数据可见性、原子性和有序性问题。不同于常规摘要的简述方式,本摘要将直接概述文章的核心内容,为读者提供一个清晰的学习路径。 ####
40 2
|
1月前
|
存储 安全 Java
什么是 Java 的内存模型?
Java内存模型(Java Memory Model, JMM)是Java虚拟机(JVM)规范的一部分,它定义了一套规则,用于指导Java程序中变量的访问和内存交互方式。
68 1
|
4月前
|
存储 编译器 C语言
【C语言篇】数据在内存中的存储(超详细)
浮点数就采⽤下⾯的规则表⽰,即指数E的真实值加上127(或1023),再将有效数字M去掉整数部分的1。
399 0
|
2月前
|
存储 C语言
数据在内存中的存储方式
本文介绍了计算机中整数和浮点数的存储方式,包括整数的原码、反码、补码,以及浮点数的IEEE754标准存储格式。同时,探讨了大小端字节序的概念及其判断方法,通过实例代码展示了这些概念的实际应用。
80 1