首页 > 系统相关 >Java内部类持有外部类导致内存泄露--原因/解决方案

Java内部类持有外部类导致内存泄露--原因/解决方案

时间:2022-10-08 11:24:00浏览次数:74  
标签:部类 Outer -- public 内存 new Java class Inner

原文网址:Java内部类持有外部类导致内存泄露--原因/解决方案_IT利刃出鞘的博客-CSDN博客

简介

说明

本文介绍Java内部类持有外部类导致内存泄露的原因以及其解决方案。

为什么内部类持有外部类会导致内存泄露?

非静态内部类会持有外部类,如果有地方引用了这个非静态内部类,会导致外部类也被引用,垃圾回收时无法回收这个外部类(即使外部类已经没有其他地方在使用了)。

解决方案

不要让其他的地方持有这个非静态内部类的引用,直接在这个非静态内部类执行业务。
将非静态内部类改为静态内部类。
内部类改为静态的之后,它所引用的对象或属性也必须是静态的,所以静态内部类无法获得外部对象的引用,只能从 JVM 的 Method Area(方法区)获取到static类型的引用。
相关网址

匿名内部类的内存泄露:Java的匿名内部类导致内存泄露--原因/解决方案_IT利刃出鞘的博客-CSDN博客

为什么要持有外部类

Java 语言中,非静态内部类的主要作用有两个:

1.当内部类只在外部类中使用时,匿名内部类可以让外部不知道它的存在,从而减少了代码的维护工作。
2.当内部类持有外部类时,它就可以直接使用外部类中的变量了,这样可以很方便的完成调用,如下代码所示:

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);
}
}

 


但是,静态内部类就无法持有外部类和其非静态字段了。比如下边这样就会报错

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);
}
}

 


断点调试

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

 

 

 

 

实例:内存泄露

简介

若内部类持有外部类的引用,对内部类的使用很多时,会导致外部类数目很多。此时,就算是外部类的数据没有被用到,外部类的数据所占空间也不会被释放。

本处在外部类存放大量的数据来模拟。

代码

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++);
}
}
}

 


测试

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

 

 

 

不会内存泄露的方案简介

内部类改为静态的之后,它所引用的对象或属性也必须是静态的,所以静态内部类无法获得外部对象的引用,只能从 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++);
}
}
}

 


测试

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

 

 

 

其他网址

Java内部类持有外部类的引用详细分析与解决方案_leunging的博客-CSDN博客_内部类持有外部类
Java防止非静态内部类内存泄漏

永远不要使用双花括号初始化实例,除非你想OOM!

 

标签:部类,Outer,--,public,内存,new,Java,class,Inner
From: https://www.cnblogs.com/cj8357475/p/16768357.html

相关文章

  • spring.jpa.open-in-view is enabled by default. Therefore, database queries may b
    在搭建微服务的过程中,发现控制台一直在报这个警告日志。本着研究的目的,查看了下源码[WARN][JpaBaseConfiguration$JpaWebConfiguration][219]:spring.jpa.open-in-viewi......
  • 手机app测试要点
    目录:一、简介1.1什么是App测试1.2  测试方法1.2.1 白盒测试1.2.2 黑盒测试1.2.3 人工测试1.2.4 自动化测试1.3  UT、IT、ST测试1.3.1 Unit Testin......
  • gateway
     gateway&nacos&ribbon负载server:port:8200spring:application:name:lc-gatewaycloud:nacos:discovery:#注册中心地址......
  • day17 110
    110平衡二叉树classSolution{publicbooleanisBalanced(TreeNoderoot){return(getHeight(root)!=-1);//返回值是一个int类型左右子树的最大高......
  • MSU Trinity Contest Petrozavodsk Winter Camp 2014
    题目列表A.MEX-QueryD.ShortEnoughTaskF.JustAnotherSequenceProblemA.ABBA题意:Solution思路:CodeconstintN=1000010;intn,......
  • 文盘Rust -- struct 中的生命周期
    最近在用rust写一个redis的数据校验工具。redis-rs中具备redis::ConnectionLiketrait,借助它可以较好的来抽象校验过程。在开发中,不免要定义struct中的某些元素为trait......
  • MUR1100-ASEMI轴向快恢复二极管MUR1100
    编辑:llMUR1100-ASEMI轴向快恢复二极管MUR1100型号:MUR1100品牌:ASEMI封装:DO-41特性:快恢复二极管正向电流:1A反向耐压:1000V恢复时间:50ns引脚数量:2芯片个数:1芯片尺寸......
  • 关于el-select做分页后切换分页显示id的问题
    刚刚用el-select做项目,数据很多使用了分页,然而当切换分页时,已选中的数据显示为id而不是name,这个地方需要增加romote属性,  还要注意,在加了clearable之后,我的可搜索功......
  • base64与中文字符串互转
    实现代码如下//字符串转base64getEncode64(str){returnbtoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,functiontoSolidBytes(match,p......
  • 带掩码的自编码器(MAE)最新的相关论文推荐
     1、HeterogeneousGraphMaskedAutoencodersYijunTian,KaiwenDong,ChunhuiZhang,ChuxuZhang,NiteshV.Chawlahttps://arxiv.org/pdf/2208.09957生......