java内部匿名类

2017-03-15  本文已影响0人  eblly

前言:
java基础的编写类的时候,会觉得很匿名内部类很神秘.普通的java文件会被编译成class,那内部类和匿名内部类呢?它们被会存在在哪个物理位置?
本文主要针对匿名内部类.

本文中有ISay和ISay2两个接口,用于测试对比.

ISay.java

public interface ISay {
     void sayHello();
 }

ISay2.java

public interface ISay2 {
     void sayHello222();

     void print();
 }

Program.java

public class Program {
    public static void main(String[] args) {
        ISay say = new ISay() {
            public void sayHello() {
                System.out.println("Hello java!");
            }
        };

        ISay say2 = new ISay() {
            public void sayHello() {
                System.out.println("Hello java!");
            }
        };

        ISay say3 = new ISay() {
            public void sayHello() {
                System.out.println("Hello java!");
            }
        };


        ISay2 say2_1 = new ISay2() {
            public void sayHello222() {
                System.out.println("Hello java!");
            }

            public void print(){
                System.out.print("print");
            }
        };
 
        say.sayHello();
        say2.sayHello();
        say2_1.sayHello222();
        say2_1.print();
    }
}

生成的class文件如下图:
ISay.java

public interface ISay {
     void sayHello();
 }

ISay2.java

public interface ISay2 {
     void sayHello222();

     void print();
 }

Program.java

public class Program {
    public static void main(String[] args) {
        ISay say = new ISay() {
            public void sayHello() {
                System.out.println("Hello java!");
            }
        };

        ISay say2 = new ISay() {
            public void sayHello() {
                System.out.println("Hello java!");
            }
        };

        ISay say3 = new ISay() {
            public void sayHello() {
                System.out.println("Hello java!");
            }
        };


        ISay2 say2_1 = new ISay2() {
            public void sayHello222() {
                System.out.println("Hello java!");
            }

            public void print(){
                System.out.print("print");
            }
        };
 
        say.sayHello();
        say2.sayHello();
        say2_1.sayHello222();
        say2_1.print();
    }
}

生成的class文件如下图:


6863841.png

发现会有Program$1.class等文件,其实这就是匿名类的class文件.

使用jclasslib查看Program.class字节码
###是注释,不是字节码的内容

new #2 <NewInterface/Program$1>                      ###对应代码ISay say = new Program$1() ,Program$1是ISay的实现。
dup                                                                                                                                                                                                       
invokespecial #3 <NewInterface/Program$1.<init>>
astore_1                                             ###存放引用在变量1中

new #4 <NewInterface/Program$2>
dup
invokespecial #5 <NewInterface/Program$2.<init>>
astore_2

new #6 <NewInterface/Program$3>
dup
invokespecial #7 <NewInterface/Program$3.<init>>
astore_3

new #8 <NewInterface/Program$4>
dup
invokespecial #9 <NewInterface/Program$4.<init>>
astore 4

aload_1                                                      ###加载本地变量1的引用,即上述的NewInterface/Program$1
invokeinterface #10 <NewInterface/ISay.sayHello> count 1     ###调用方法,对象是本地变量1的实例。

aload_2
invokeinterface #10 <NewInterface/ISay.sayHello> count 1

aload 4                                                      ###加载本地变量4的引用,即上述的NewInterface/Program$4
invokeinterface #11 <NewInterface/ISay2.sayHello222> count 1 ###调用方法,对象是本地变量4的实例。

aload 4
invokeinterface #12 <NewInterface/ISay2.print> count 1
return

内部类跟匿名内部类的class生成规则差不多,只不过匿名内部类是由编译器命令,内部类是直接指定.
例如:Program中有一个内部类叫Book,那么生成的class文件是Program$Book.class

上一篇下一篇

猜你喜欢

热点阅读