Java内部类是一种编译器现象与虚拟机无关

内部类是一种编译器现象与虚拟机无关。编译器将内部类翻译为用$分隔外部类名和内部类名的常规类文件,虚拟机对此并无所知。
使用javap -private OuterClass$InnerClass。javap这个工具确实挺不错的,对分析字节码和源码都有很大的帮助。
可以看出详细的内部类源码清单,其中包括了编译器自动添加的部分:

public class Outer
{
 public class Inner
 {
 }
}
当内部类是非静态内部类时相应的内部类的详细源码如下:
Compiled from "Outer.java"
public class Outer$Inner extends java.lang.Object{
    private final Outer this$0;  //编译器自动在内部类里面添加了指向外部类对象的引用
    public Outer$Inner(Outer){}  //内部类的构造函数默认有一个外部类对象作为参数。
}

当内部类是静态内部类时:
Compiled from "Outer.java"
public class Outer$Inner extends java.lang.Object{
    public Outer$Inner(){}  //没有了对外部类对象的引用
}


匿名类也是一样的道理

class Outer
{

  public static void f(){

     new Runnable(){ 

        public void run(){

        }

     };

  }

   public void g(){

      new Runnable(){

         public void run(){

         }

      };

   }

}


class Outer$1 implement Runnable{

   public Outer$1(){

   }

   public void run(){

   }

}


class Outer$2 implement Runnable{

   private Outer outer;

   public Outer$2(Outer outer){

   }

   public void run(){

   }


}


// 关于匿名类中只能引用方法中final类型的局部变量

package org.taobao;


public class Test7{


public static void main(String[] args) {


}


public void f(final int a, final int b){




Runnable runnable = new Runnable() {
@Override
public void run() {
xxx(a, b);
}
};


}


public void xxx(int a, int b){

}


}

class org.taobao.Test7$1 extends java.lang.Object implements java.lang.Runnable{
    final org.taobao.Test7 this$0;
    private final int val$a;
    private final int val$b;
    org.taobao.Test7$1(org.taobao.Test7, int, int); // 创建匿名类对象时,就会把用到的上层方法的final局部变量当做参数传入构造器,成为匿名类的一个private的成员变量
    public void run();
}







你可能感兴趣的:(Java内部类是一种编译器现象与虚拟机无关)