Java中,关于匿名内部类调用外部类方法的疑惑

发布于 2022-09-04 20:27:43 字数 1121 浏览 32 评论 0

今天学习了内部类的知识,知道内部类是可以持有外部类的this,从而在内部类中可以使用OuterClass.this.medthod()来引用相应外部类方法。但是我写出下代码,可以运行,然而其中的调用逻辑我不是很明白,望赐教!

public class test {
        public void report(){
            System.out.println("I'm invoked!");
        }
        public void perform(){
            new Speaker().handleAction(new Action(){
                @Override
                public 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???

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

不再让梦枯萎 2022-09-11 20:27:44
        public void perform(){
            new Speaker().handleAction(new Action(){
                @Override
                public void action() {
                    report();//???为什么能调用report??
                }
            });
        }

new Speaker()不是匿名内部类,它有确切的类名Speaker
new 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));
        }

所以匿名内部类持有外部类的引用,且可以调用外部类的方法

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文