常见的内存泄漏及其解决方案
内存泄漏是Java开发中一个常见且令人头疼的问题,即使在使用垃圾回收机制的Java中,也无法完全避免内存泄漏的出现。当对象不再需要时却仍然占据着内存,导致内存使用量不断增加,最终可能导致 OutOfMemoryError
。本文将深入探讨Java中常见的内存泄漏及其解决方案,附带详细的代码示例,帮助你更好地理解和解决内存泄漏问题。
1. 常见的内存泄漏场景
- 静态集合类引起的内存泄漏
- 未关闭的IO资源
- 监听器和回调的非预期持有
- ThreadLocal引起的内存泄漏
- 自定义类加载器引起的内存泄漏
2. 静态集合类引起的内存泄漏
静态集合类(如 HashMap
, ArrayList
等)在应用程序生命周期内是静态的,如果没有适当地移除不再需要的对象,会导致这些对象无法被垃圾回收,从而引起内存泄漏。
示例代码:
import java.util.HashMap;
import java.util.Map;public class StaticCollectionLeak {private static final Map<Integer, String> cache = new HashMap<>();public void addToCache(int id, String value) {cache.put(id, value);}public static void main(String[] args) {StaticCollectionLeak leak = new StaticCollectionLeak();for (int i = 0; i < 100000; i++) {leak.addToCache(i, "value" + i);}// 内存使用量会不断增加}
}
解决方案:
确保及时移除不再需要的对象,或者使用 WeakHashMap
替代 HashMap
。
import java.util.WeakHashMap;
import java.util.Map;public class StaticCollectionSolution {private static final Map<Integer, String> cache = new WeakHashMap<>();public void addToCache(int id, String value) {cache.put(id, value);}public static void main(String[] args) {StaticCollectionSolution solution = new StaticCollectionSolution();for (int i = 0; i < 100000; i++) {solution.addToCache(i, "value" + i);}// 内存使用量不会持续增加}
}
3. 未关闭的IO资源
未关闭的 InputStream
或 OutputStream
等IO资源,会导致内存泄漏。
示例代码:
import java.io.FileInputStream;
import java.io.IOException;public class UnclosedIOLeak {public void readFile(String filePath) throws IOException {FileInputStream fis = new FileInputStream(filePath);// Do something with fis// 未关闭FileInputStream}public static void main(String[] args) {UnclosedIOLeak leak = new UnclosedIOLeak();try {leak.readFile("somefile.txt");} catch (IOException e) {e.printStackTrace();}}
}
解决方案:
使用 try-with-resources
确保IO资源被自动关闭。
import java.io.FileInputStream;
import java.io.IOException;public class ClosedIOSolution {public void readFile(String filePath) throws IOException {try (FileInputStream fis = new FileInputStream(filePath)) {// Do something with fis} // FileInputStream将在这里被自动关闭}public static void main(String[] args) {ClosedIOSolution solution = new ClosedIOSolution();try {solution.readFile("somefile.txt");} catch (IOException e) {e.printStackTrace();}}
}
4. 监听器和回调的非预期持有
注册的监听器或回调在不再需要时如果未被删除,会导致内存泄漏。
示例代码:
import java.util.ArrayList;
import java.util.List;public class ListenerLeak {private final List<Runnable> listeners = new ArrayList<>();public void registerListener(Runnable listener) {listeners.add(listener);}// 没有方法来移除监听器public static void main(String[] args) {ListenerLeak leak = new ListenerLeak();leak.registerListener(() -> System.out.println("Listener 1"));leak.registerListener(() -> System.out.println("Listener 2"));}
}
解决方案:
提供移除监听器的方法,并在不需要时及时移除。
import java.util.ArrayList;
import java.util.List;public class ListenerSolution {private final List<Runnable> listeners = new ArrayList<>();public void registerListener(Runnable listener) {listeners.add(listener);}public void unregisterListener(Runnable listener) {listeners.remove(listener);}public static void main(String[] args) {ListenerSolution solution = new ListenerSolution();Runnable listener1 = () -> System.out.println("Listener 1");Runnable listener2 = () -> System.out.println("Listener 2");solution.registerListener(listener1);solution.registerListener(listener2);// 移除监听器,避免内存泄漏solution.unregisterListener(listener1);solution.unregisterListener(listener2);}
}
5. ThreadLocal引起的内存泄漏
ThreadLocal
对象如果不及时移除,会导致内存泄漏,尤其是在使用线程池的情况下。
示例代码:
public class ThreadLocalLeak {private static final ThreadLocal<byte[]> threadLocal = ThreadLocal.withInitial(() -> new byte[1024 * 1024]);public static void main(String[] args) {threadLocal.get(); // 分配1MB内存// 未调用remove方法,导致内存泄漏}
}
解决方案:
在不需要时调用 ThreadLocal.remove()
方法移除对象。
public class ThreadLocalSolution {private static final ThreadLocal<byte[]> threadLocal = ThreadLocal.withInitial(() -> new byte[1024 * 1024]);public static void main(String[] args) {try {threadLocal.get(); // 分配1MB内存} finally {threadLocal.remove(); // 在使用后移除,避免内存泄漏}}
}
6. 自定义类加载器引起的内存泄漏
自定义类加载器如果未能正确卸载类,会导致内存泄漏。
示例代码:
public class CustomClassLoaderLeak {public static void main(String[] args) throws Exception {while (true) {CustomClassLoader loader = new CustomClassLoader();Class<?> clazz = loader.loadClass("LeakClass");Object instance = clazz.getDeclaredConstructor().newInstance();// 每次循环都会创建新的类加载器,但旧的类加载器未被释放}}static class CustomClassLoader extends ClassLoader {// 自定义类加载器实现}
}
解决方案:
确保自定义类加载器不再使用时,可以被垃圾回收器回收。
public class CustomClassLoaderSolution {public static void main(String[] args) throws Exception {while (true) {CustomClassLoader loader = new CustomClassLoader();Class<?> clazz = loader.loadClass("LeakClass");Object instance = clazz.getDeclaredConstructor().newInstance();// 使loader对象可以被回收loader = null;System.gc(); // 提示GC进行垃圾回收}}static class CustomClassLoader extends ClassLoader {// 自定义类加载器实现}
}
结论
Java中的内存泄漏虽然不如C/C++那样常见,但仍然是需要关注的问题。通过识别常见的内存泄漏场景并采取适当的解决方案,可以有效地减少和避免内存泄漏的发生。希望本文提供的示例和解决方案能够帮助你在实际开发中更好地处理内存泄漏问题。
相关文章:
常见的内存泄漏及其解决方案
内存泄漏是Java开发中一个常见且令人头疼的问题,即使在使用垃圾回收机制的Java中,也无法完全避免内存泄漏的出现。当对象不再需要时却仍然占据着内存,导致内存使用量不断增加,最终可能导致 OutOfMemoryError。本文将深入探讨Java中…...
SQLSERVER 触发器记录表某个字段更新记录
想要记录该字段的原值和现有值,触发器写法: CREATE TRIGGER tr_UpdateEmployeeDepartment ON Employees AFTER UPDATE AS BEGINSET NOCOUNT ON; -- 避免多余的计数消息IF UPDATE(Department) -- 检查是否更新了 Department 字段BEGININSERT INTO Update…...

现代前端架构介绍(第一部分):App是如何由不同的构建块构成的
远离JavaScript疲劳和框架大战,了解真正重要的东西 几周前,我的同事们对我们的前端架构、代码结构和面临的挑战很感兴趣。在做了几次关于如何构建可扩展且健壮的前端的演讲后,我觉得把它们都总结一下并与社区分享我们的策略是一个不错的主意。…...
Android 11 关于按键拦截/按键事件处理分享
系统在frameworks/base/services/core/java/com/android/server/policy/PhoneWindowManager.java处理按键事件,不管是物理按键还是 SystemUI的nav_bar上的虚拟按键(使用了KeyEvent类中的,比如:KeyEvent.KEYCODE_VOLUME_UP). 主要注意的有两个函数: interceptKeyBef…...

最新TomatoIDC开源虚拟主机销售系统源码/有插件系统模块+模版系统
源码简介: 最新TomatoIDC开源虚拟主机销售系统源码,它有一个方便扩展的插件和模版系统模块,使用实用。 TomatoIDC,一款遵循GPL3.0协议的开源虚拟主机销售系统,不仅有着可以轻松扩展的插件系统和模版系统,…...

简单的docker学习 第4章docker容器
第4章 Docker容器 4.1 容器基础 4.1.1 容器启动流程 通过 docker run 命令可以启动运行一个容器。该命令在执行时首先会在本地查找指定的镜像,如果找到了,则直接启动,否则会到镜像中心查找。如果镜像中心存在该镜像,则会下载到…...
PHP中如何声明数组
数组是一种数据结构,用于存储一系列的值或对象,这些值或对象可以通过索引(或键)来访问。在PHP中,数组是一种复合类型的数据结构,可以存储多个值,这些值可以是整型、字符串、布尔值,甚…...

JavaScript前端面试题——fetch
什么是fetch? fetch:fetch是浏览器内置的api,用于发送网络请求 ajax&axios&fetch的关系 ajax:ajax 是一种基于原生 JavaScript 的异步请求技术。它使用 XMLHttpRequest 对象来发送请求和接收响应。 axios:…...

在Qt中获取Windows中进程的PID
主要是用到了系统自带的工具【tasklist.exe】 利用 QProcess调用这个tasklist有一点坑,已经在代码中指出了。 指定为csv格式输出的话,在后处理时比较方便。 QList<quint64> listProcessIdentifier(QString processName) {QProcess process;QStrin…...

8.1-java+tomcat环境的配置+代理
一、回顾 1.安装nodejs,这是一个jdk一样的软件运行环境 yum -y list installed|grep epel yum -y install nodejs node -v 2.下载对应的nodejs软件npm yum -y install npm npm -v npm set config .....淘宝镜像 3.安装vue/cli command line interface 命令行…...
gorm框架实现基本的增删改查
连接数据库 package mainimport ("github.com/jinzhu/gorm"_ "github.com/jinzhu/gorm/dialects/mysql" )func main() {db, err : gorm.Open("mysql","root:roottcp(127.0.0.1:3306)/test?charsetutf8mb4&parseTimeTrue&locLocal…...

AUTOSAR介绍
1、AUTOSAR架构介绍 AUTOSAR(AUTomotive Open System ARchitecture,汽车开放系统架构)是汽车和软件行业领先公司的全球合作联盟,为智能移动开发和建立标准化的软件框架以及开放的E/E系统架构。考虑到目前和未来市场中不同的汽车E/E架构,AUTOS…...

10. 计算机网络HTTP协议
1. 前言 无论是作为后端开发、前端开发、测试开发程序员或者是运维人员,在面试过程中,大概率都会被问到 HTTP 协议相关题目。 因为伴随着 2010 年之后移动互联网在全世界的高速发展,各种各样的浏览器(Chrome、FireFox、Safari 等)层出不穷,也诞生了诸多服务端开发的语言…...

“职场中,不要和上司作对”,真的很重要吗?你认同这句话吗?
在职场上,领导对下属的期望永远都只有两个字,不是忠诚,也不是能力,而是省心。 领导对下属的要求就是别让我操心。 在职场中,通常面临的首要问题就是如何与领导相处。 把职场中的前辈当作老师来尊重,你尊…...

可视化目标检测算法推理部署(一)Gradio的UI设计
引言 在先前RT-DETR模型的学习过程中,博主自己使用Flask框架搭建了一个用于模型推理的小案例: FlaskRT-DETR模型推理 在这个过程中,博主需要学习Flask、HTML等相关内容,并且博主做出的页面还很丑,那么,是…...

【PyTorch】基于YOLO的多目标检测项目(一)
【PyTorch】基于YOLO的多目标检测项目(一) 【PyTorch】基于YOLO的多目标检测项目(二) 目标检测是对图像中的现有目标进行定位和分类的过程。识别的对象在图像中显示有边界框。一般的目标检测方法有两种:基于区域提议的…...

spring boot 实现 Stream 钉钉事件订阅
1: 参考链接 https://open.dingtalk.com/document/orgapp/develop-stream-mode-push-server 2:钉钉开放平台订阅配置 配置之后运行一下上面提供的链接 里面的main方法,验证通道 3:订阅启动方式 EventListenerThread eventListenerThrea…...

基于 Rough.js 的 Vue 散点图绘制
本文由ScriptEcho平台提供技术支持 项目地址:传送门 基于 Rough.js 的 Vue 散点图绘制 应用场景 本代码展示了如何使用 Rough.js 库在 Vue 应用程序中绘制散点图。Rough.js 是一个轻量级 JavaScript 库,用于创建具有手绘风格的可视化效果。散点图是一…...
【c++】用c++指针传递来模拟“靶向治疗”
一:源码: #include <iostream>void targetedTherapy(bool* flag) {if (*flag == false) {*flag = true;} }int main() {//代表一系列癌细胞//true为健康细胞 false为癌变细胞bool cancerCell[7] = {true, false, true, true, true, true, false};for (int i = 0; i &…...

如何开启idea中的断言功能?
目录 一、什么是断言? 二、Java断言的语法 三、开启断言 一、什么是断言? 断言(assert)是 Java 中的一条语句,一种在程序中的逻辑(如一个结果为真或假的逻辑判断式),目的是验证软…...
【Linux】C语言执行shell指令
在C语言中执行Shell指令 在C语言中,有几种方法可以执行Shell指令: 1. 使用system()函数 这是最简单的方法,包含在stdlib.h头文件中: #include <stdlib.h>int main() {system("ls -l"); // 执行ls -l命令retu…...
服务器硬防的应用场景都有哪些?
服务器硬防是指一种通过硬件设备层面的安全措施来防御服务器系统受到网络攻击的方式,避免服务器受到各种恶意攻击和网络威胁,那么,服务器硬防通常都会应用在哪些场景当中呢? 硬防服务器中一般会配备入侵检测系统和预防系统&#x…...

NLP学习路线图(二十三):长短期记忆网络(LSTM)
在自然语言处理(NLP)领域,我们时刻面临着处理序列数据的核心挑战。无论是理解句子的结构、分析文本的情感,还是实现语言的翻译,都需要模型能够捕捉词语之间依时序产生的复杂依赖关系。传统的神经网络结构在处理这种序列依赖时显得力不从心,而循环神经网络(RNN) 曾被视为…...

nnUNet V2修改网络——暴力替换网络为UNet++
更换前,要用nnUNet V2跑通所用数据集,证明nnUNet V2、数据集、运行环境等没有问题 阅读nnU-Net V2 的 U-Net结构,初步了解要修改的网络,知己知彼,修改起来才能游刃有余。 U-Net存在两个局限,一是网络的最佳深度因应用场景而异,这取决于任务的难度和可用于训练的标注数…...

【iOS】 Block再学习
iOS Block再学习 文章目录 iOS Block再学习前言Block的三种类型__ NSGlobalBlock____ NSMallocBlock____ NSStackBlock__小结 Block底层分析Block的结构捕获自由变量捕获全局(静态)变量捕获静态变量__block修饰符forwarding指针 Block的copy时机block作为函数返回值将block赋给…...

Python异步编程:深入理解协程的原理与实践指南
💝💝💝欢迎莅临我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 持续学习,不断…...

作为点的对象CenterNet论文阅读
摘要 检测器将图像中的物体表示为轴对齐的边界框。大多数成功的目标检测方法都会枚举几乎完整的潜在目标位置列表,并对每一个位置进行分类。这种做法既浪费又低效,并且需要额外的后处理。在本文中,我们采取了不同的方法。我们将物体建模为单…...
软件工程教学评价
王海林老师您好。 您的《软件工程》课程成功地将宏观的理论与具体的实践相结合。上半学期的理论教学中,您通过丰富的实例,将“高内聚低耦合”、SOLID原则等抽象概念解释得十分透彻,让这些理论不再是停留在纸面的名词,而是可以指导…...

Ansys Maxwell:线圈和磁体的静磁 3D 分析
本博客展示了如何在 Ansys Maxwell 中执行静磁 3D 分析,以计算载流线圈和永磁体之间相互作用产生的扭矩。在这个例子中,线圈中的电流产生一个沿 Y 轴指向的磁场,而永磁体沿 X 轴被磁化。这种配置导致围绕 Z 轴的扭矩。分步工作流程包括构建几…...

自建 dnslog 回显平台:渗透测试场景下的隐蔽回显利器
🔍 背景介绍 在渗透测试与红队评估过程中,DNS 外带(DNS Exfiltration) 是一种常见且隐蔽的通信通道。由于多数目标环境默认具备外网 DNS 解析能力,即便在 无回显、无文件上传权限 的条件下,仍可通过 DNS 请…...