oop - Java中,关于匿名内部类调用外部类方法的疑惑
问题描述
今天学习了内部类的知识,知道内部类是可以持有外部类的this,从而在内部类中可以使用OuterClass.this.medthod()来引用相应外部类方法。但是我写出下代码,可以运行,然而其中的调用逻辑我不是很明白,望赐教!
public class test {public void report(){ System.out.println('I’m invoked!');}public void perform(){ new Speaker().handleAction(new Action(){@Overridepublic void action() { report();//???为什么能调用report??} });}public static void main(String[] args) { new test().perform();//测试代码} } class Speaker{void handleAction(Action act){ act.action();} } interface Action{void action(); }
其中设计是这样的,test对象调用perform方法,该方法其中新建一个Speaker匿名类对象,该对象调用了其handleAction方法,该方法的参数是一个Action接口,接口需要重写action抽象方法。我使用了属于test的report方法。输出是正常的。
那么我想知道,test对象的方法中有一个匿名类的局部对象,局部对象参数是一个实现接口的匿名类,为什么在这个匿名类中可以调用report?它持有test.this指针吗?我理解中,new Speaker().handleAction(new Action(){....这里面的实现逻辑和test.this一点关系都没有,也没有必要持有test.this???
问题解答
回答1:public void perform(){ new Speaker().handleAction(new Action(){@Overridepublic void action() { report();//???为什么能调用report??} });}
new Speaker()不是匿名内部类,它有确切的类名Speakernew Action(){}是匿名内部类,会在编译的时候给它一个类名(我们假定它的类名叫Test$1,你可以打印this.getClass()看看)看出两者的区别了吗?匿名内部类多出了个{}。由于new Action(){}是在test对象的作用域里被创建的,所以它的外部类是Test。
匿名内部类也有构造器,而且和普通类的构造器有点不一样,编译的时候会在匿名内部类的构造器的参数列表之前再插入一个参数,这个参数是外部类的对象的引用,编译之后这个类长这样:
Test$1 implements Action { final T this$0; Test$1(T this$0){this.this$0 = this$0; }@Override public void action() {this$0.report(); }}
new Action(){...}实际上是创建了Test$1,并且通过构造器把test对象引用传给Test$1
public void perform(){ new Speaker().handleAction(new Test$1(this));}
所以匿名内部类持有外部类的引用,且可以调用外部类的方法