Java 内部类有坑。。100 % 内存泄露!
来源:https://knife.blog.csdn.net/article/details/124946774
今天给大家分享一种,Java内部类使用不当导致的内存泄露问题,最终导致内存溢出!希望能够帮助到大家!
简介
「说明」
本文介绍 Java 内部类持有外部类导致内存泄露的原因以及其解决方案。
「为什么内部类持有外部类会导致内存泄露?」
非静态内部类会持有外部类,如果有地方引用了这个非静态内部类,会导致外部类也被引用,垃圾回收时无法回收这个外部类(即使外部类已经没有其他地方在使用了)。
「解决方案」
- 不要让其他的地方持有这个非静态内部类的引用,直接在这个非静态内部类执行业务。
- 将非静态内部类改为静态内部类。内部类改为静态的之后,它所引用的对象或属性也必须是静态的,所以静态内部类无法获得外部对象的引用,只能从 JVM 的 Method Area(方法区)获取到static类型的引用。
为什么要持有外部类
Java 语言中,非静态内部类的主要作用有两个:
- 当内部类只在外部类中使用时,匿名内部类可以让外部不知道它的存在,从而减少了代码的维护工作。
- 当内部类持有外部类时,它就可以直接使用外部类中的变量了,这样可以很方便的完成调用,如下代码所示:
package org.example.a;
class Outer{
private String outerName = "Tony";
class Inner{
private String name;
public Inner() {
this.name = outerName;
}
}
Inner createInner() {
return new Inner();
}
}
public class Demo {
public static void main(String[] args) {
Outer.Inner inner = new Outer().createInner();
System.out.println(inner);
}
}
但是,静态内部类就无法持有外部类和其非静态字段了。另外,最新 Java 面试题整理:https://www.javastack.cn/mst/
比如下边这样就会报错
package org.example.a;
class Outer{
private String outerName = "Tony";
static class Inner{
private String name;
public Inner() {
this.name = outerName;
}
}
Inner createInner() {
return new Inner();
}
}
public class Demo {
public static void main(String[] args) {
Outer.Inner inner = new Outer().createInner();
System.out.println(inner);
}
}
报错:
实例:持有外部类
「代码」
package org.example.a;
class Outer{
class Inner {
}
Inner createInner() {
return new Inner();
}
}
public class Demo {
public static void main(String[] args) {
Outer.Inner inner = new Outer().createInner();
System.out.println(inner);
}
}
「断点调试」
可以看到:内部类持有外部类的对象的引用,是以“this$0”这个字段来保存的。
实例:不持有外部类
「代码」
package org.example.a;
class Outer{
static class Inner {
}
Inner createInner() {
return new Inner();
}
}
public class Demo {
public static void main(String[] args) {
Outer.Inner inner = new Outer().createInner();
System.out.println(inner);
}
}
更多 Java 教程及示例:https://github.com/javastacks/javastack
「断点调试」
可以发现:内部类不再持有外部类了。
实例:内存泄露
「简介」
若内部类持有外部类的引用,对内部类的使用很多时,会导致外部类数目很多。此时,就算是外部类的数据没有被用到,外部类的数据所占空间也不会被释放。
本处在外部类存放大量的数据来模拟。
「代码」
package org.example.a;
import java.util.ArrayList;
import java.util.List;
class Outer{
private int[] data;
public Outer(int size) {
this.data = new int[size];
}
class Innner{
}
Innner createInner() {
return new Innner();
}
}
public class Demo {
public static void main(String[] args) {
List<Object> list = new ArrayList<>();
int counter = 0;
while (true) {
list.add(new Outer(100000).createInner());
System.out.println(counter++);
}
}
}
「测试」
可以看到:运行了八千多次的时候就内存溢出了。
我换了一台 mac 电脑,4000 多就内存溢出了。
不会内存泄露的方案
「简介」
内部类改为静态的之后,它所引用的对象或属性也必须是静态的,所以静态内部类无法获得外部对象的引用,只能从 JVM 的 Method Area(方法区)获取到 static 类型的引用。
「代码」
package org.example.a;
import java.util.ArrayList;
import java.util.List;
class Outer{
private int[] data;
public Outer(int size) {
this.data = new int[size];
}
static class Inner {
}
Inner createInner() {
return new Inner();
}
}
public class Demo {
public static void main(String[] args) {
List<Object> list = new ArrayList<>();
int counter = 0;
while (true) {
list.add(new Outer(100000).createInner());
System.out.println(counter++);
}
}
}
「测试」
可以发现:循环了四十多万次都没有内存溢出。
近期热文推荐:
1.1,000+ 道 Java面试题及答案整理(2022最新版)
4.别再写满屏的爆爆爆炸类了,试试装饰器模式,这才是优雅的方式!!
觉得不错,别忘了随手点赞+转发哦!
相关文章
- java中图片文件的传输及显示(Socket以及ServerSocket演示)
- Java 正则表达式格式化时间显示
- java nio 映射文件内容到内存
- Java数组与内存控制
- 深入理解JMM(Java内存模型) --(一)
- 如何诊断 Java 中的内存泄露
- Java Enum
- java入门学习(十二)运算语句 if switch
- Java虚拟机的内存结构
- [二]Java虚拟机 jvm内存结构 运行时数据内存 class文件与jvm内存结构的映射 jvm数据类型 虚拟机栈 方法区 堆 含义
- Java内存模型
- Java动态修改任意用户Session,终极解决方案,锋哥出品
- JAVA基础-栈与堆,static、final修饰符、内部类和Java内存分配
- Java虚拟机内存区域堆(heap)的管理
- 【Java AWT 图形界面编程】FileDialog 对话框 ( 打开文件 | 保存文件 | 构造函数 | 获取文件路径 | 获取文件名称 | 代码示例 )
- Android内存优化3 了解java GC 垃圾回收机制1
- Java 使用URLEncoder.encode和URLDecoder.decode编解码(utf-8)中文及特殊字符
- 1093 字符串A+B(JAVA)
- 实战突击: Java Web项目整合开发(PDF)
- 解决idea启动项目报错:Unable to open debugger port(127.0.0.1:60157):java.net.SocketException"socket closed