CSDN本来想用“{{”秀一波,结果却导致了内存溢出!
本文插图
作者 | 磊哥 来源 | Java中文社群责编 | 王晓曼 生活中的尴尬无处不在 , 有时候你只是想简单的装一把 , 但某些“老同志”总是在不经意之间 , 给你无情的一脚 , 踹得你简直无法呼吸 。但谁让咱年轻呢?吃亏要趁早 , 前路会更好 。喝了这口温热的鸡汤 , 咱们来聊聊是怎么回事 。事情是这样的 , 在一个不大不小的项目中 , 小王写下了这段代码:Map<String, String> map = new HashMap {{ put("map1", "value1"); put("map2", "value2"); put("map3", "value3");}};map.forEach((k, v) -> { System.out.println("key:" + k + " value:" + v);});本来是用它来替代下面这段代码的:Map<String, String> map = new HashMap;map.put("map1", "value1");map.put("map2", "value2");map.put("map3", "value3");map.forEach((k, v) -> { System.out.println("key:" + k + " value:" + v);});两块代码的执行结果也是完全一样的:
- key:map3value:value3
- key:map2value:value2
- key:map1value:value1
本文插图
我们使用 Idea 打开 DoubleBracket$1.class 文件发现:import java.util.HashMap;class DoubleBracket$1 extends HashMap { DoubleBracket$1(DoubleBracket var1) { this.this$0 = var1; this.put("map1", "value1"); this.put("map2", "value2"); }}此时我们可以确认 , 它就是一个匿名内部类 。 那么问题来了 , 匿名内部类为什么会导致内存溢出呢?匿名内部类的“锅”在 Java 语言中非静态内部类会持有外部类的引用 , 从而导致 GC 无法回收这部分代码的引用 , 以至于造成内存溢出 。思考 1:为什么要持有外部类?这个就要从匿名内部类的设计说起了 , 在 Java 语言中 , 非静态匿名内部类的主要作用有两个 。1、当匿名内部类只在外部类(主类)中使用时 , 匿名内部类可以让外部不知道它的存在 , 从而减少了代码的维护工作 。2、当匿名内部类持有外部类时 , 它就可以直接使用外部类中的变量了 , 这样可以很方便的完成调用 , 如下代码所示:public class DoubleBracket { private static String userName = "磊哥"; public static void main(String args) throws NoSuchFieldException, IllegalAccessException { Map<String, String> map = new HashMap {{ put("map1", "value1"); put("map2", "value2"); put("map3", "value3"); put(userName, userName); }}; }}从上述代码可以看出在 HashMap 的方法内部 , 可以直接使用外部类的变量 userName 。思考 2:它是怎么持有外部类的?关于匿名内部类是如何持久外部对象的 , 我们可以通过查看匿名内部类的字节码得知 , 我们使用 javap -c DoubleBracket\$1.class 命令进行查看 , 其中 $1 为以匿名类的字节码 , 字节码的内容如下:javap -c DoubleBracket\$1.classCompiled from "DoubleBracket.java"class com.example.DoubleBracket$1 extends java.util.HashMap { final com.example.DoubleBracket this$0;com.example.DoubleBracket$1(com.example.DoubleBracket); Code: 0: aload_0 1: aload_1 2: putfield #1 // Field this$0:Lcom/example/DoubleBracket; 5: aload_0 6: invokespecial #7 // Method java/util/HashMap."<init>":V 9: aload_0 10: ldc #13 // String map1 12: ldc #15 // String value1 14: invokevirtual #17 // Method put:(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object; 17: pop 18: aload_0 19: ldc #21 // String map2 21: ldc #23 // String value2 23: invokevirtual26: pop 27: return}其中 , 关键代码的在 putfield 这一行 , 此行表示有一个对 DoubleBracket 的引用被存入到 this$0 中 , 也就是说这个匿名内部类持有了外部类的引用 。如果您觉得以上字节码不够直观 , 没关系 , 我们用下面的实际的代码来证明一下:import java.lang.reflect.Field;import java.util.HashMap;import java.util.Map;public class DoubleBracket { public static void main(String args) throws NoSuchFieldException, IllegalAccessException { Map map = new DoubleBracket.createMap; // 获取一个类的所有字段 Field field = map.getClass.getDeclaredField("this$0"); // 设置允许方法私有的 private 修饰的变量 field.setAccessible(true); System.out.println(field.get(map).getClass); } public Map createMap { // 双花括号初始化 Map map = new HashMap {{ put("map1", "value1"); put("map2", "value2"); put("map3", "value3"); }}; return map; }}当我们开启调试模式时 , 可以看出 map 中持有了外部对象 DoubleBracket , 如下图所示:
推荐阅读
- 再看清你本来的面目,说你原来是这样的人,我对你的感觉大概就是
- 王者荣耀|玩个1V1单挑要啥观众呢?小兵超级兵前来助阵!草丛版本来袭
- 骑猪赏日出|结果尴尬的没法收场,搞笑GIF:小姐姐本来想作秀
- 三言财经 AirPods,彻底占据你的耳朵,苹果想用
- 丈夫|丈夫意外去世,妻子想用冷冻胚胎生孩子却遭拒!法院判了
- 双眼皮男生|本来想走楼梯的, 可走到电梯旁却听到...,开心一刻:去医院挂号,
- CSDN23 岁创业,28 岁成为福布斯亚洲青年领袖,这个“刷脸的男人”有点牛
- 大司马解说|新版本来临前再冲一次分!,LOL云顶之弈:战地枪手强势登场
- 广之美|有以下5款面霜大家秋冬用了还想用,网红人气面霜!
- 不明白小姐|周迅穿廉价衣服走机场,素颜本来就显老了,也不会打扮年轻点
