内部类是一种编译器现象与虚拟机无关。编译器将内部类翻译为用$分隔外部类名和内部类名的常规类文件,虚拟机对此并无所知。
使用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();
}