Java内部类持有外部类的引用详细分析与解决方案

Posted ZhangSeachal的笔记

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java内部类持有外部类的引用详细分析与解决方案相关的知识,希望对你有一定的参考价值。

在Java中内部类的定义与使用一般为成员内部类与匿名内部类,他们的对象都会隐式持有外部类对象的引用,影响外部类对象的回收。

GC只会回收没有被引用或者根集不可到达的对象(取决于GC算法),内部类在生命周期内始终持有外部类的对象的引用,造成外部类的对象始终不满足GC的回收条件,反映在内存上就是内存泄露。(如,android中Activity的内存泄露)

解决方案为

1.将内部类定义为static

2.用static的变量引用匿名内部类的实例

测试代码

  1. class Outer {  
  2.     class Inner {  
  3.         public String publicString = "Inner.publicString";  
  4.     }  
  5.   
  6.     Other anonymousOther = new Other() {  
  7.         public String publicString = "Anonymous Other.publicString";  
  8.     };  
  9.     public Other getAnonymousOther() {  
  10.         return anonymousOther;  
  11.     }  
  12.   
  13.     Other Other = new Other();  
  14.     public Other getOther() {  
  15.         return Other;  
  16.     }  
  17. }  
  18.   
  19. class Other {  
  20.     public String publicString = "Other.publicString";  
  21. }  

调用代码

  1. public static void main(String args[]) {  
  2.         printField(new Outer().new Inner());  
  3.         System.out.println("\t");  
  4.         printField(new Outer().getAnonymousOther());  
  5.         System.out.println("\t");  
  6.         printField(new Outer().getOther());  
  7.     }  

 

测试结果

  1. Class: at.miao.Outer$Inner  
  2. 变量: publicString 值为 Inner.publicString  
  3. 变量: this$0 值为 [email protected]  
  4.       
  5. Class: at.miao.Outer$1  
  6. 变量: publicString 值为 Anonymous Other.publicString  
  7. 变量: this$0 值为 [email protected]  
  8. Class: at.miao.Other  
  9. 变量: publicString 值为 Other.publicString  


可以看到内部类与匿名内部类的实例都有一个外部类类型的名为this$0的变量指向了外部类的对象。

加上static之后,代码为

  1. class Outer {  
  2.     static class Inner {  
  3.         public String publicString = "Inner.publicString";  
  4.     }  
  5.   
  6.     static Other anonymousOther = new Other() {  
  7.         public String publicString = "Anonymous Other.publicString";  
  8.     };  
  9.   
  10.     public Other getAnonymousOther() {  
  11.         return anonymousOther;  
  12.     }  
  13.   
  14.     Other Other = new Other();  
  15.     public Other getOther() {  
  16.         return Other;  
  17.     }  
  18. }  
  19.   
  20. class Other {  
  21.     public String publicString = "Other.publicString";  
  22. }  

调用代码

  1. public static void main(String args[]) {  
  2.         printField(new Outer.Inner());  
  3.         System.out.println("\t");  
  4.         printField(new Outer().getAnonymousOther());  
  5.         System.out.println("\t");  
  6.         printField(new Outer().getOther());  
  7.     }  

 

测试结果

  1. Class: at.miao.Outer$Inner  
  2. 变量: publicString 值为 Inner.publicString  
  3.       
  4. Class: at.miao.Outer$1  
  5. 变量: publicString 值为 Anonymous Other.publicString  
  6.       
  7. Class: at.miao.Other  
  8. 变量: publicString 值为 Other.publicString  

可以看到静态内部类实例、static引用的匿名内部类的实例未引用外部类的实例。


以上是关于Java内部类持有外部类的引用详细分析与解决方案的主要内容,如果未能解决你的问题,请参考以下文章

Android 开发必备知识点及面试题汇总(Android+Java+算法+性能优化+四大组件……)

note(java)

内部类和静态内部类有什么区别?

Handler机制与原理

Handler机制与原理

Handler机制与原理