Redian新闻
>
Java内部类有坑,100%内存泄露!

Java内部类有坑,100%内存泄露!

公众号新闻

点击上方“芋道源码”,选择“设为星标

管她前浪,还是后浪?

能浪的浪,才是好浪!

每天 10:33 更新文章,每天掉亿点点头发...

源码精品专栏

 
来源:knife.blog.csdn.net

简介

「说明」

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

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

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

「解决方案」

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

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://gitee.com/zhijiantianya/ruoyi-vue-pro
  • 视频教程:https://doc.iocoder.cn/video/

为什么要持有外部类

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

报错:

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://gitee.com/zhijiantianya/yudao-cloud
  • 视频教程:https://doc.iocoder.cn/video/

实例:持有外部类

「代码」

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

「测试」

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

换了一台 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++);
        }
    }
}

「测试」

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

以上,希望能对大家在使用内部类时会有所帮助。



欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢

已在知识星球更新源码解析如下:

最近更新《芋道 SpringBoot 2.X 入门》系列,已经 101 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。

提供近 3W 行代码的 SpringBoot 示例,以及超 4W 行代码的电商微服务项目。

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。

文章有帮助的话,在看,转发吧。

谢谢支持哟 (*^__^*)

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
RansomEXX勒索软件团伙声称入侵法拉利公司并泄露内部文件【信息安全三分钟】2022.10.05澳洲知名科技公司数据泄露!服务客户超1000家,包括维州政府重大!澳洲Medibank用户信息再遭泄露!黑客扬言:不给钱“周五继续”!全球最强智算来了:12000000000000000000(别数了18个0)FLOPS!美国国债,破31000000000000美元!突发!Telstra也翻车了!数万员工信息泄露!Optus210万用户需要换证件,华人要注意这些事!炸裂!亚麻OOD面试标准答案内部全泄露…市值蒸发$1000000000000!Amazon被爆酝酿史上最血腥大裁员!现代人与已灭绝古人类有何差异?解读2022年诺贝尔生理学或医学奖澳人注意!Optus群发短信!收到代表信息泄露!需尽快换驾照!不用健康告知的重疾险?有没有坑《消失的孩子》,又一部类似《隐秘的角落》的爆款?大快人心!亚麻系统设计面试标准答案内部全泄露…每秒1000000000000000000次运算!Cerebras新超级计算机打造世界最大1350万核心AI集群Medibank更多隐私数据被泄露!总理也卷入!避免受骗,记好这五点!江歌遗体照遭泄露!埋了6年的真相彻底反转?江秋莲,终于还是瞒不住了……又是数据泄露!澳临床实验室系统被黑,患者信息被贴上暗网!22.3万人受影响!深刻理解 | 以通信方式共享内存,不要以共享内存方式通信华人速查!Optus刚遭受史上最大规模黑客攻击,上千万人个人信息泄露!包括护照、姓名、生日、电话、电子邮件、家庭地址等近30万客户信息或泄露!知名企业道歉OPPO新机现身 Geekbench跑分,16GB 内存、80W 快充鲍里斯的手机号码被公开15年后,英媒昨日曝料:政府数据遭到泄露!劲爆!亚麻coding面试标准答案内部全泄露…维州知名科技公司数据泄露!服务客户超1000家,包括维州政府10名纽约华人背地干这事儿!1年里200名同胞被坑,涉案$1800万Costco 的红油抄手,Aldi的美味青口和几款小菜Chrome再次优化内存占用,新增内存释放开关教育随笔(105)高考文言文备考之六刚刚,毕马威内部绝密计划泄露…最近做的家常饭西雅圖華人超市見聞【一路有你】《想某人》2019深刻理解 | 以通信方式共享内存,不要以共享内存方式通信澳国防部也遭“黑手”!通信平台遭勒索软件攻击,或有数据泄露!970万澳人中招!Medibank遭受特大信息泄露!华人该如何应对?你需要知道这些!
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。