1. 程式人生 > >Java內部類持有外部類的引用詳細分析與解決方案

Java內部類持有外部類的引用詳細分析與解決方案

調用 lai urn star keyword inner android get sta

在Java中內部類的定義與使用一般為成員內部類與匿名內部類,他們的對象都會隱式持有外部類對象的引用,影響外部類對象的回收。

GC只會回收沒有被引用或者根集不可到達的對象(取決於GC算法),內部類在生命周期內始終持有外部類的對象的引用,造成外部類的對象始終不滿足GC的回收條件,反映在內存上就是內存泄露。(如,Android中Activity的內存泄露)

解決方案為

1.將內部類定義為static

2.用static的變量引用匿名內部類的實例

測試代碼

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

調用代碼

  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. Class: at.miao.Outer$1
  5. 變量: publicString 值為 Anonymous Other.publicString
  6. 變量: this$0 值為 [email protected]
  7. Class: at.miao.Other
  8. 變量: publicString 值為 Other.publicString


可以看到內部類與匿名內部類的實例都有一個外部類類型的名為this$0的變量指向了外部類的對象。

加上static之後,代碼為

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

調用代碼

  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. Class: at.miao.Outer$1
  4. 變量: publicString 值為 Anonymous Other.publicString
  5. Class: at.miao.Other
  6. 變量: publicString 值為 Other.publicString

可以看到靜態內部類實例、static引用的匿名內部類的實例未引用外部類的實例。

Java內部類持有外部類的引用詳細分析與解決方案