ToB企服应用市场:ToB评测及商务社交产业平台

标题: 注意避坑!Java 内部类持有外部类会导致内存泄露。。。 [打印本页]

作者: 冬雨财经    时间: 2023-9-14 02:24
标题: 注意避坑!Java 内部类持有外部类会导致内存泄露。。。
简介

说明

本文介绍 Java 内部类持有外部类导致内存泄露的原因以及其解决方案。
为什么内部类持有外部类会导致内存泄露
非静态内部类会持有外部类,如果有地方引用了这个非静态内部类,会导致外部类也被引用,垃圾回收时无法回收这个外部类(即使外部类已经没有其他地方在使用了)。
解决方案

不要让其他的地方持有这个非静态内部类的引用,直接在这个非静态内部类执行业务。
将非静态内部类改为静态内部类。内部类改为静态的之后,它所引用的对象或属性也必须是静态的,所以静态内部类无法获得外部对象的引用,只能从 JVM 的 Method Area(方法区)获取到static类型的引用。
推荐一个开源免费的 Spring Boot 实战项目:
https://github.com/javastacks/spring-boot-best-practice
为什么要持有外部类

Java 语言中,非静态内部类的主要作用有两个:
  1. package org.example.a;
  2. class Outer{
  3.     private String outerName = "Tony";
  4.     class Inner{
  5.         private String name;
  6.         public Inner() {
  7.             this.name = outerName;
  8.         }
  9.     }
  10.     Inner createInner() {
  11.         return new Inner();
  12.     }
  13. }
  14. public class Demo {
  15.     public static void main(String[] args) {
  16.         Outer.Inner inner = new Outer().createInner();
  17.         System.out.println(inner);
  18.     }
  19. }
复制代码
但是,静态内部类就无法持有外部类和其非静态字段了。
比如下边这样就会报错:
  1. package org.example.a;
  2. class Outer{
  3.     private String outerName = "Tony";
  4.     static class Inner{
  5.         private String name;
  6.         public Inner() {
  7.             this.name = outerName;
  8.         }
  9.     }
  10.     Inner createInner() {
  11.         return new Inner();
  12.     }
  13. }
  14. public class Demo {
  15.     public static void main(String[] args) {
  16.         Outer.Inner inner = new Outer().createInner();
  17.         System.out.println(inner);
  18.     }
  19. }
复制代码
报错:

实例:持有外部类

代码
  1. package org.example.a;
  2. class Outer{
  3.     class Inner {
  4.     }
  5.     Inner createInner() {
  6.         return new Inner();
  7.     }
  8. }
  9. public class Demo {
  10.     public static void main(String[] args) {
  11.         Outer.Inner inner = new Outer().createInner();
  12.         System.out.println(inner);
  13.     }
  14. }
复制代码
断点调试

可以看到:内部类持有外部类的对象的引用,是以“this$0”这个字段来保存的。

实例:不持有外部类

推荐一个开源免费的 Spring Boot 实战项目:
https://github.com/javastacks/spring-boot-best-practice
  1. package org.example.a;
  2. class Outer{
  3.     static class Inner {
  4.     }
  5.     Inner createInner() {
  6.         return new Inner();
  7.     }
  8. }
  9. public class Demo {
  10.     public static void main(String[] args) {
  11.         Outer.Inner inner = new Outer().createInner();
  12.         System.out.println(inner);
  13.     }
  14. }
复制代码
断点调试

可以发现:内部类不再持有外部类了。

实例:内存泄露

简介

若内部类持有外部类的引用,对内部类的使用很多时,会导致外部类数目很多。此时,就算是外部类的数据没有被用到,外部类的数据所占空间也不会被释放。
本处在外部类存放大量的数据来模拟。
代码
  1. package org.example.a;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. class Outer{
  5.     private int[] data;
  6.     public Outer(int size) {
  7.         this.data = new int[size];
  8.     }
  9.     class Innner{
  10.     }
  11.     Innner createInner() {
  12.         return new Innner();
  13.     }
  14. }
  15. public class Demo {
  16.     public static void main(String[] args) {
  17.         List<Object> list = new ArrayList<>();
  18.         int counter = 0;
  19.         while (true) {
  20.             list.add(new Outer(100000).createInner());
  21.             System.out.println(counter++);
  22.         }
  23.     }
  24. }
复制代码
测试

可以看到:运行了八千多次的时候就内存溢出了。

我换了一台 mac 电脑,4000 多就内存溢出了。

不会内存泄露的方案

简介

内部类改为静态的之后,它所引用的对象或属性也必须是静态的,所以静态内部类无法获得外部对象的引用,只能从 JVM 的 Method Area(方法区)获取到 static 类型的引用。
代码
  1. package org.example.a;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. class Outer{
  5.     private int[] data;
  6.     public Outer(int size) {
  7.         this.data = new int[size];
  8.     }
  9.     static class Inner {
  10.     }
  11.     Inner createInner() {
  12.         return new Inner();
  13.     }
  14. }
  15. public class Demo {
  16.     public static void main(String[] args) {
  17.         List<Object> list = new ArrayList<>();
  18.         int counter = 0;
  19.         while (true) {
  20.             list.add(new Outer(100000).createInner());
  21.             System.out.println(counter++);
  22.         }
  23.     }
  24. }
复制代码
测试

可以发现:循环了四十多万次都没有内存溢出。

来源:blog.csdn.net/feiying0canglang/article/details/121108201
近期热文推荐:
1.1,000+ 道 Java面试题及答案整理(2022最新版)
2.劲爆!Java 协程要来了。。。
3.Spring Boot 2.x 教程,太全了!
4.别再写满屏的爆爆爆炸类了,试试装饰器模式,这才是优雅的方式!!
5.《Java开发手册(嵩山版)》最新发布,速速下载!
觉得不错,别忘了随手点赞+转发哦!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!




欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/) Powered by Discuz! X3.4