当前位置: 首页 > news >正文

常见的内存泄漏及其解决方案

内存泄漏是Java开发中一个常见且令人头疼的问题,即使在使用垃圾回收机制的Java中,也无法完全避免内存泄漏的出现。当对象不再需要时却仍然占据着内存,导致内存使用量不断增加,最终可能导致 OutOfMemoryError。本文将深入探讨Java中常见的内存泄漏及其解决方案,附带详细的代码示例,帮助你更好地理解和解决内存泄漏问题。

1. 常见的内存泄漏场景
  1. 静态集合类引起的内存泄漏
  2. 未关闭的IO资源
  3. 监听器和回调的非预期持有
  4. ThreadLocal引起的内存泄漏
  5. 自定义类加载器引起的内存泄漏
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资源

未关闭的 InputStreamOutputStream 等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开发中一个常见且令人头疼的问题&#xff0c;即使在使用垃圾回收机制的Java中&#xff0c;也无法完全避免内存泄漏的出现。当对象不再需要时却仍然占据着内存&#xff0c;导致内存使用量不断增加&#xff0c;最终可能导致 OutOfMemoryError。本文将深入探讨Java中…...

SQLSERVER 触发器记录表某个字段更新记录

想要记录该字段的原值和现有值&#xff0c;触发器写法&#xff1a; CREATE TRIGGER tr_UpdateEmployeeDepartment ON Employees AFTER UPDATE AS BEGINSET NOCOUNT ON; -- 避免多余的计数消息IF UPDATE(Department) -- 检查是否更新了 Department 字段BEGININSERT INTO Update…...

现代前端架构介绍(第一部分):App是如何由不同的构建块构成的

远离JavaScript疲劳和框架大战&#xff0c;了解真正重要的东西 几周前&#xff0c;我的同事们对我们的前端架构、代码结构和面临的挑战很感兴趣。在做了几次关于如何构建可扩展且健壮的前端的演讲后&#xff0c;我觉得把它们都总结一下并与社区分享我们的策略是一个不错的主意。…...

Android 11 关于按键拦截/按键事件处理分享

系统在frameworks/base/services/core/java/com/android/server/policy/PhoneWindowManager.java处理按键事件,不管是物理按键还是 SystemUI的nav_bar上的虚拟按键(使用了KeyEvent类中的,比如:KeyEvent.KEYCODE_VOLUME_UP). 主要注意的有两个函数&#xff1a; interceptKeyBef…...

最新TomatoIDC开源虚拟主机销售系统源码/有插件系统模块+模版系统

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

简单的docker学习 第4章docker容器

第4章 Docker容器 4.1 容器基础 4.1.1 容器启动流程 通过 docker run 命令可以启动运行一个容器。该命令在执行时首先会在本地查找指定的镜像&#xff0c;如果找到了&#xff0c;则直接启动&#xff0c;否则会到镜像中心查找。如果镜像中心存在该镜像&#xff0c;则会下载到…...

PHP中如何声明数组

数组是一种数据结构&#xff0c;用于存储一系列的值或对象&#xff0c;这些值或对象可以通过索引&#xff08;或键&#xff09;来访问。在PHP中&#xff0c;数组是一种复合类型的数据结构&#xff0c;可以存储多个值&#xff0c;这些值可以是整型、字符串、布尔值&#xff0c;甚…...

JavaScript前端面试题——fetch

什么是fetch&#xff1f; fetch&#xff1a;fetch是浏览器内置的api&#xff0c;用于发送网络请求 ajax&axios&fetch的关系 ajax&#xff1a;ajax 是一种基于原生 JavaScript 的异步请求技术。它使用 XMLHttpRequest 对象来发送请求和接收响应。 axios&#xff1a;…...

在Qt中获取Windows中进程的PID

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

8.1-java+tomcat环境的配置+代理

一、回顾 1.安装nodejs&#xff0c;这是一个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&#xff0c;汽车开放系统架构)是汽车和软件行业领先公司的全球合作联盟&#xff0c;为智能移动开发和建立标准化的软件框架以及开放的E/E系统架构。考虑到目前和未来市场中不同的汽车E/E架构&#xff0c;AUTOS…...

10. 计算机网络HTTP协议

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

“职场中,不要和上司作对”,真的很重要吗?你认同这句话吗?

在职场上&#xff0c;领导对下属的期望永远都只有两个字&#xff0c;不是忠诚&#xff0c;也不是能力&#xff0c;而是省心。 领导对下属的要求就是别让我操心。 在职场中&#xff0c;通常面临的首要问题就是如何与领导相处。 把职场中的前辈当作老师来尊重&#xff0c;你尊…...

可视化目标检测算法推理部署(一)Gradio的UI设计

引言 在先前RT-DETR模型的学习过程中&#xff0c;博主自己使用Flask框架搭建了一个用于模型推理的小案例&#xff1a; FlaskRT-DETR模型推理 在这个过程中&#xff0c;博主需要学习Flask、HTML等相关内容&#xff0c;并且博主做出的页面还很丑&#xff0c;那么&#xff0c;是…...

【PyTorch】基于YOLO的多目标检测项目(一)

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

spring boot 实现 Stream 钉钉事件订阅

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

基于 Rough.js 的 Vue 散点图绘制

本文由ScriptEcho平台提供技术支持 项目地址&#xff1a;传送门 基于 Rough.js 的 Vue 散点图绘制 应用场景 本代码展示了如何使用 Rough.js 库在 Vue 应用程序中绘制散点图。Rough.js 是一个轻量级 JavaScript 库&#xff0c;用于创建具有手绘风格的可视化效果。散点图是一…...

【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中的断言功能?

目录 一、什么是断言&#xff1f; 二、Java断言的语法 三、开启断言 一、什么是断言&#xff1f; 断言&#xff08;assert&#xff09;是 Java 中的一条语句&#xff0c;一种在程序中的逻辑&#xff08;如一个结果为真或假的逻辑判断式&#xff09;&#xff0c;目的是验证软…...

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…...

<6>-MySQL表的增删查改

目录 一&#xff0c;create&#xff08;创建表&#xff09; 二&#xff0c;retrieve&#xff08;查询表&#xff09; 1&#xff0c;select列 2&#xff0c;where条件 三&#xff0c;update&#xff08;更新表&#xff09; 四&#xff0c;delete&#xff08;删除表&#xf…...

MFC内存泄露

1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...

学校招生小程序源码介绍

基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码&#xff0c;专为学校招生场景量身打造&#xff0c;功能实用且操作便捷。 从技术架构来看&#xff0c;ThinkPHP提供稳定可靠的后台服务&#xff0c;FastAdmin加速开发流程&#xff0c;UniApp则保障小程序在多端有良好的兼…...

智能在线客服平台:数字化时代企业连接用户的 AI 中枢

随着互联网技术的飞速发展&#xff0c;消费者期望能够随时随地与企业进行交流。在线客服平台作为连接企业与客户的重要桥梁&#xff0c;不仅优化了客户体验&#xff0c;还提升了企业的服务效率和市场竞争力。本文将探讨在线客服平台的重要性、技术进展、实际应用&#xff0c;并…...

Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; &#x1f680; AI篇持续更新中&#xff01;&#xff08;长期更新&#xff09; 目前2025年06月05日更新到&#xff1a; AI炼丹日志-28 - Aud…...

springboot整合VUE之在线教育管理系统简介

可以学习到的技能 学会常用技术栈的使用 独立开发项目 学会前端的开发流程 学会后端的开发流程 学会数据库的设计 学会前后端接口调用方式 学会多模块之间的关联 学会数据的处理 适用人群 在校学生&#xff0c;小白用户&#xff0c;想学习知识的 有点基础&#xff0c;想要通过项…...

接口自动化测试:HttpRunner基础

相关文档 HttpRunner V3.x中文文档 HttpRunner 用户指南 使用HttpRunner 3.x实现接口自动化测试 HttpRunner介绍 HttpRunner 是一个开源的 API 测试工具&#xff0c;支持 HTTP(S)/HTTP2/WebSocket/RPC 等网络协议&#xff0c;涵盖接口测试、性能测试、数字体验监测等测试类型…...

逻辑回归暴力训练预测金融欺诈

简述 「使用逻辑回归暴力预测金融欺诈&#xff0c;并不断增加特征维度持续测试」的做法&#xff0c;体现了一种逐步建模与迭代验证的实验思路&#xff0c;在金融欺诈检测中非常有价值&#xff0c;本文作为一篇回顾性记录了早年间公司给某行做反欺诈预测用到的技术和思路。百度…...

苹果AI眼镜:从“工具”到“社交姿态”的范式革命——重新定义AI交互入口的未来机会

在2025年的AI硬件浪潮中,苹果AI眼镜(Apple Glasses)正在引发一场关于“人机交互形态”的深度思考。它并非简单地替代AirPods或Apple Watch,而是开辟了一个全新的、日常可接受的AI入口。其核心价值不在于功能的堆叠,而在于如何通过形态设计打破社交壁垒,成为用户“全天佩戴…...