我们一起聊聊 Java 内存泄漏

开发 前端
使用Java编写程序时,我们使用new关键字创建对象。而且我们还不需要专门在对象使用完成后去释放其占用的内存,这是因为Java有专门的垃圾回收器来负责删除不需要的对象。

Java内存泄漏一直Java程序中最常见的问题之一,它会导致内存溢出,最终导致程序崩溃。我们可能对内存泄漏很熟悉,但又不是那么熟悉,真的遇到事故的时候,内存泄漏问题排查起来却也没有那么容易。本篇就再次梳理一下Java内存泄漏的那些事。

前言

使用Java编写程序时,我们使用new关键字创建对象。而且我们还不需要专门在对象使用完成后去释放其占用的内存,这是因为Java有专门的垃圾回收器来负责删除不需要的对象。只要不被使用的对象有垃圾回收器回收,那么程序会处于正常运行的状态,但是垃圾回收器无法删除那些不被使用的对象时,我们的Java程序则可能发生了内存泄漏。

内存泄漏是什么

内存泄漏指的是JVM中某些不再需要使用的对象,仍然存活于JVM中而不能及时释放而导致内存空间的浪费。Java中内存泄漏的原因有多种,这些众多的因素会导致Java程序产生不同类型的内存泄漏,随着时间的推移,内存泄漏会使程序增加额外的内存资源占用,从而导致程序性能下降。

垃圾回收器会回收长时间没有引用的对象,但是它不会回收那些还存在引用的对象,这就是产生内存泄漏的原因。

所以为了防止内存泄漏,程序设计之初就需要考虑去释放那些不使用的内存空间,而开发人员也应当时刻考虑内存泄漏的可能性,并增加一些测试和检测避免内存泄漏。

堆和栈的内存泄漏

Java中,我们可能会遇到栈内存泄露和堆内存泄漏。

其中堆内存泄漏是由于创建后的对象一直存在于堆中,不再需要的对象其引用一直没有被移除。这些无用的对象会慢慢占用内存,最后导致内存溢出。

栈内存泄漏由于方法不断被调用,但是一直没有退出方法。这种情况可能发生在无限循环或递归掉用时,最终导致栈内存溢出。

内存泄漏的原因

Java中内存泄漏主要是因为不能正确释放不需要的资源,长生命周期对象持有短生命周期对象的引用。

  • 静态字段

静态字段引起的内存泄漏比较常见,如果某个不需要的类中含有静态字段,那么就会造成内存泄漏。单例模式中如果持有其他的类引用就会造成内存泄漏,静态集合如HashMap,LinkedList等持有的一些对象没有及时释放等。

  • Thread Local

threadlocal引用一个对象使用完成后并没有被及时remove掉,线程一直存活的情况下(使用线程池时)就会发生内存泄漏。

大多时候内存泄漏都是由于开发人员的代码错误导致的,要防止这种内存泄漏,就需要编写必要的代码来配合垃圾回收器释放资源。

避免Java内存泄漏的一些最佳实践

  • 使用最新稳定版本的Java
  • 尽量减少使用静态变量,使用完之后及时赋值 null,移除引用
  • 明确对象的有效作用域,尽量缩小对象的作用域。局部变量回收会很快。
  • 减少长生命周期对象持有短生命周期的引用
  • 各种连接应该及时关闭(数据库连接,网络,IO等)
  • 使用内存泄漏检测工具如MAT,Visual VM,jprofile 等
  • 避免在代码中使用System.gc()
  • 避免使用内部类

内存泄漏很难定位并修复,但是我们可以遵循以下几个步骤去定位并修复:

  • 确定是否存在内存泄漏,启用详细的GC跟踪。
  • 使用一些第三方插件进行分析(jprofile  Visual VM等)
  • 检查调用堆栈是否有未释放的引用(分析GC状态)
  • 找出对象没有被垃圾回收的原因
  • 编写代码手动删除此类对象
责任编辑:武晓燕 来源: Java技术指北
相关推荐

2023-03-26 23:47:32

Go内存模型

2023-08-10 08:28:46

网络编程通信

2023-08-04 08:20:56

DockerfileDocker工具

2023-09-10 21:42:31

2023-06-30 08:18:51

敏捷开发模式

2022-05-24 08:21:16

数据安全API

2021-08-27 07:06:10

IOJava抽象

2024-02-20 21:34:16

循环GolangGo

2022-12-06 08:12:11

Java关键字

2023-11-10 08:04:43

Java 17Java 11JDK

2022-09-22 08:06:29

计算机平板微信

2024-07-26 09:47:28

2022-10-08 00:00:05

SQL机制结构

2023-07-24 09:41:08

自动驾驶技术交通

2021-08-12 07:49:24

mysql

2022-02-23 08:41:58

NATIPv4IPv6

2024-11-28 09:57:50

C#事件发布器

2022-10-28 07:27:17

Netty异步Future

2022-04-06 08:23:57

指针函数代码

2022-01-04 12:08:46

设计接口
点赞
收藏

51CTO技术栈公众号